Skip to content

Instantly share code, notes, and snippets.

@1wheel
Last active March 22, 2017 07:30
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save 1wheel/d921d0d42eb30be865c31a7d0a92eece to your computer and use it in GitHub Desktop.
Save 1wheel/d921d0d42eb30be865c31a7d0a92eece to your computer and use it in GitHub Desktop.
characters-2140
var names = [
"Mutt and Jeff",
"",
"Inspector Gen",
"Franklin",
"Vlade",
"Citizen",
"Amelia",
"Charlotte",
"Stefan and Roberto"
]
var radius = 1
var length = 100
var nameToAngle = d3.scaleOrdinal()
.domain(names)
.range(d3.range(0, Math.PI*2, Math.PI*2/names.length))
var zScale = d3.scaleLinear()
.domain([0, 1111347])
.range([0, length])
d3.json('chapters.json', function(err, res){
chapters = res
links = []
var prevLength = 0
chapters.forEach(function(chapter){
chapter.links.forEach(function(d){
d.tIndex = d.index + prevLength
d.narrator = chapter.name
d.tName = d.name
.replace('Mutt', 'Mutt and Jeff')
.replace('Jeff', 'Mutt and Jeff')
.replace('Gen', 'Inspector Gen')
.replace('Stefan', 'Stefan and Roberto')
.replace('Roberto', 'Stefan and Roberto')
d.pos = [
Math.cos(nameToAngle(d.narrator))*radius, Math.sin(nameToAngle(d.narrator))*radius,
]
})
links = links.concat(chapter.links)
chapter.start = prevLength
chapter.end = prevLength += chapter.length
})
// Create an empty scene
var scene = new THREE.Scene();
// Create a basic perspective camera
var camera = new THREE.PerspectiveCamera( 75, window.innerWidth/window.innerHeight, 0.1, 10000 );
camera.position.z = length*1.01;
// Create a renderer with Antialiasing
var renderer = new THREE.WebGLRenderer({antialias:true});
// Configure renderer clear color
renderer.setClearColor("#2E2B40");
// Configure renderer size
renderer.setSize( window.innerWidth, window.innerHeight );
// Append Renderer to DOM
d3.selectAll('canvas').remove()
document.body.appendChild( renderer.domElement );
var material = new THREE.MeshBasicMaterial( { color: "#ffffff" } );
chapters.forEach(function(d){
var geometry = new THREE.BoxGeometry(.05, 0.05, zScale(d.length));
var bar02 = new THREE.Mesh( geometry, material );
bar02.position.x = Math.cos(nameToAngle(d.name))*radius;
bar02.position.y = Math.sin(nameToAngle(d.name))*radius;
bar02.position.z = zScale(d.start)
scene.add( bar02 );
})
// line geometry
var lineGeometry = new THREE.Geometry();
links.forEach(function(d){
var z = zScale(d.tIndex)
var x0 = Math.cos(nameToAngle(d.narrator))*radius;
var y0 = Math.sin(nameToAngle(d.narrator))*radius;
var vertexPos = new THREE.Vector3(x0, y0, z);
lineGeometry.vertices.push(vertexPos);
var x1 = Math.cos(nameToAngle(d.tName))*radius;
var y1 = Math.sin(nameToAngle(d.tName))*radius;
var vertexPos = new THREE.Vector3(x1, y1, z);
lineGeometry.vertices.push(vertexPos);
})
// line object
var lineMaterial = new THREE.LineBasicMaterial({color:0xffff00});
var lineObject = new THREE.Line(lineGeometry, lineMaterial, THREE.LineSegments);
lineObject.position.set
scene.add(lineObject);
// Render Loop
if (window.timer) window.timer.stop()
window.timer = d3.timer(function() {
// bar02.rotation.z+=0.001;
camera.position.z -= length/5000
// Render the scene
renderer.render(scene, camera);
})
})
[{"chapter":1,"name":"Mutt and Jeff","length":8954,"links":[{"index":1112,"str":"“Mutt, hang with me","name":"Mutt"},{"index":5869,"str":"Mutt laughs too, to see his friend so amused","name":"Mutt"},{"index":7056,"str":"Mutt moves his lips when he reads","name":"Mutt"},{"index":8206,"str":"“Shit indeed,” Mutt says","name":"Mutt"},{"index":8533,"str":"“It’s like you said before,” Mutt points out","name":"Mutt"},{"index":8840,"str":"Mutt and Jeff regard each other","name":"Mutt"},{"index":8908,"str":"Mutt nods","name":"Mutt"},{"index":5563,"str":"“Please Jeff? You’re sounding scary","name":"Jeff"},{"index":5818,"str":"Jeff laughs till he puts his forehead on the table","name":"Jeff"},{"index":6314,"str":"Jeff says, “So look, we know how to get into these systems, we know how to write code, we are the best coders in the world","name":"Jeff"},{"index":7590,"str":"Jeff nods","name":"Jeff"},{"index":8110,"str":"Jeff reads his computer screen","name":"Jeff"},{"index":8359,"str":"” Jeff frowns","name":"Jeff"},{"index":8840,"str":"Mutt and Jeff regard each other","name":"Jeff"},{"index":8873,"str":"Jeff thumbs toward the staircases","name":"Jeff"}]},{"chapter":2,"name":"","length":42,"links":[]},{"chapter":3,"name":"Inspector Gen","length":10310,"links":[{"index":0,"str":"Inspector Gen Octaviasdottir sat in her office, late again, slumped in her chair, trying to muster the energy to get up and go home","name":"Gen"},{"index":441,"str":"Wide-set intelligent eyes, now observing Gen sharply; expressive mouth","name":"Gen"},{"index":554,"str":"But she looked as tired as Gen felt","name":"Gen"},{"index":783,"str":"“I thought you looked familiar,” Gen said","name":"Gen"},{"index":1140,"str":"Inspector Gen had recently inherited her apartment from her mother, and she paid little attention to how the building was run","name":"Gen"},{"index":1932,"str":"“Are you going back to the Met now?” Gen asked","name":"Gen"},{"index":2046,"str":"” Gen turned to Olmstead","name":"Gen"},{"index":2298,"str":"Armstrong headed toward the elevators and looked surprised when Inspector Gen suggested they walk instead","name":"Gen"},{"index":2468,"str":"“Nothing direct,” Gen explained, “but you can take the one from here to Bellevue, and then go downstairs and cross diagonally and then head west on the Twenty-third Skyline","name":"Gen"},{"index":2944,"str":"Gen tried to remember anything from the Met’s frequent bulletins","name":"Gen"},{"index":3019,"str":"But she was pretty sure this woman had been the chairperson of the co-op’s executive board since Gen had moved in to take care of her mom, which suggested three or four terms in office, not something most people would volunteer for","name":"Gen"},{"index":4542,"str":"Gen led her to the elevator in Bellevue’s northwest annex that would take them down to the skybridge that ran west from building to building on the north side of Twenty-third","name":"Gen"},{"index":4718,"str":"Most skybridges still ran either north-south or east-west, forcing what Gen called knight moves","name":"Gen"},{"index":4815,"str":"Recently some new higher skybridges made bishop moves, which pleased Gen, as she played the find-the-shortest-route game when getting around the city, played it with a gamer’s passion","name":"Gen"},{"index":5554,"str":"Gen began to regret suggesting the walk","name":"Gen"},{"index":5982,"str":"Gen nodded as Charlotte sketched this history","name":"Gen"},{"index":6295,"str":"As they pushed through the triple doors Gen nodded to the guard on duty, Manuel, who was chatting to his wrist and looked startled to see them","name":"Gen"},{"index":6439,"str":"Gen looked back out the glass doors; down at canal level the bathtub ring exposed by low tide was blackish green","name":"Gen"},{"index":6954,"str":"“So these missing guys lived on the farm floor?” Gen asked","name":"Gen"},{"index":7449,"str":"Gen spent very little time in the farm, but she did like to cook once in a while, so she put in an hour a month to be able to make a claim","name":"Gen"},{"index":8143,"str":"Gen wandered the farm, looking for anomalies","name":"Gen"},{"index":8753,"str":"The situation had some oddities that were making Gen curious","name":"Gen"},{"index":9325,"str":"Gen had seen him around in the usual way","name":"Gen"},{"index":9618,"str":"Gen asked questions, watched him describe what had happened from his perspective","name":"Gen"},{"index":9800,"str":"Depressed people did not usually engage in criminal conspiracies, Gen had long ago concluded","name":"Gen"},{"index":1773,"str":"Vlade, the building super, would have reported it, but I was coming here anyway to represent a client, so I filed the report and then asked to speak to you","name":"Vlade"},{"index":3685,"str":"It takes attention, but Vlade and his people do that","name":"Vlade"},{"index":7844,"str":"They lost their previous housing somehow, so Vlade set up a hotello they brought with them","name":"Vlade"},{"index":9367,"str":"Vlade Marovich","name":"Vlade"},{"index":10181,"str":"I’ll want a list of all the people who work for you on the building,” she said to Vlade","name":"Vlade"},{"index":652,"str":"“Hi, I’m Charlotte Armstrong,” the woman said","name":"Charlotte"},{"index":5982,"str":"Gen nodded as Charlotte sketched this history","name":"Charlotte"}]},{"chapter":4,"name":"","length":503,"links":[]},{"chapter":5,"name":"Franklin","length":21616,"links":[{"index":11124,"str":"“My Franklin Franklin!” he said as always","name":"Franklin"},{"index":19556,"str":"“Franklin Garr","name":"Franklin"},{"index":19573,"str":"“Franklin? Not Frank?”","name":"Franklin"},{"index":19596,"str":"“Franklin","name":"Franklin"},{"index":19651,"str":"“Ben Franklin","name":"Franklin"}]},{"chapter":6,"name":"","length":619,"links":[]},{"chapter":7,"name":"Vlade","length":7522,"links":[{"index":4461,"str":"“Ralph Muttchopf and Jeff Rosen","name":"Mutt"},{"index":4461,"str":"“Ralph Muttchopf and Jeff Rosen","name":"Jeff"},{"index":3848,"str":"Then Inspector Gen showed up","name":"Gen"},{"index":0,"str":"Vlade’s little apartment was located at the back of the boathouse office, down a set of broad stairs","name":"Vlade"},{"index":230,"str":"Vlade didn’t mind this","name":"Vlade"},{"index":702,"str":"Vlade was currently working with the team from the local waterproofing association that had caissoned the Madison Square side of the building to reseal the building’s wall and the old sidewalk","name":"Vlade"},{"index":1365,"str":"But Vlade kept thinking things were as good as they could get","name":"Vlade"},{"index":1484,"str":"That’s you, Vlade","name":"Vlade"},{"index":1726,"str":"Lots of complaints to share about all kinds of things, such as being paid in wetbits and blocknecklaces, which some called torcs, as they were basically forms of indenture to the building, a fancy version of room and board—people went on and on, but despite all the moaning they were lively and helped keep Vlade out of the depths","name":"Vlade"},{"index":3416,"str":"It was true there were people who needed a fight a day to satisfy some awful itch, but Vlade made them find it elsewhere","name":"Vlade"},{"index":4438,"str":"Vlade nodded unhappily","name":"Vlade"},{"index":4781,"str":"Vlade shrugged","name":"Vlade"},{"index":5002,"str":"Vlade felt a little relief as he watched her walk away","name":"Vlade"},{"index":6267,"str":"Vlade, we’re not in any trouble today,” the shorter one called up through the slats of the dock","name":"Vlade"},{"index":6410,"str":"“So come up here and tell me what you want,” Vlade said, still distracted by the policewoman","name":"Vlade"},{"index":6736,"str":"“I think soon,” Vlade said","name":"Vlade"},{"index":6861,"str":"Vlade? We heard she saw grizzly bears","name":"Vlade"},{"index":6902,"str":"“You just want to see her naked butt,” Vlade pointed out","name":"Vlade"},{"index":6981,"str":"Vlade nodded","name":"Vlade"},{"index":7367,"str":"“That’s what he meant,” the shorter one said, snatching the box from Vlade before he might change his mind","name":"Vlade"},{"index":6636,"str":"The shorter one said, “We were wondering if you know when Amelia Black is going to get back here","name":"Amelia"},{"index":4865,"str":"Or so I am led to understand by Charlotte","name":"Charlotte"}]},{"chapter":8,"name":"","length":410,"links":[]},{"chapter":9,"name":"Citizen","length":11180,"links":[]},{"chapter":10,"name":"","length":395,"links":[]},{"chapter":11,"name":"Amelia","length":17236,"links":[{"index":13537,"str":"Amelia Black, Inspector Gen Octaviasdottir","name":"Gen"},{"index":13772,"str":"“Six years ago,” Gen said","name":"Gen"},{"index":13897,"str":"Gen shrugged","name":"Gen"},{"index":14215,"str":"As she dug in, Vlade and Gen talked about people Amelia didn’t know","name":"Gen"},{"index":14639,"str":"“High profile,” Gen noted","name":"Gen"},{"index":16492,"str":"“Polar bears,” Inspector Gen said","name":"Gen"},{"index":11121,"str":"“Vlade, I’m coming in from the west, are you ready for me?”","name":"Vlade"},{"index":11182,"str":"“Always,” Vlade replied after a short pause","name":"Vlade"},{"index":13349,"str":"Vlade was at his table by the window overlooking the bacino, sitting with a woman Amelia didn’t know","name":"Vlade"},{"index":13451,"str":"Amelia approached, and Vlade introduced them: “Forty-twenty, this is Twenty-forty","name":"Vlade"},{"index":14215,"str":"As she dug in, Vlade and Gen talked about people Amelia didn’t know","name":"Vlade"},{"index":14406,"str":"“What’s up?” Vlade said","name":"Vlade"},{"index":14666,"str":"“Where can you move them?” Vlade asked","name":"Vlade"},{"index":15960,"str":"Amelia ended the call and looked up at Vlade and the policewoman","name":"Vlade"},{"index":16053,"str":"“The defenders?” Vlade asked","name":"Vlade"},{"index":16442,"str":"“No one agrees with anything,” Vlade said darkly","name":"Vlade"},{"index":16872,"str":"“Your specialty,” Vlade noted","name":"Vlade"},{"index":0,"str":"One of Amelia Black’s favorite flyways ran from Montana east over the Missouri River and south toward the Ozarks, then east into Kentucky and through the Delaware Gap and across the pine barrens, briefly out to sea and up to New York","name":"Amelia"},{"index":1026,"str":"Amelia’s cloud show was about assisting the migration of endangered species to ecozones where they were more likely to survive the changed climate, so the sight of all the nearly unoccupied land passing below, for hour after hour, was fairly common for her, but nevertheless always encouraging to see","name":"Amelia"},{"index":3562,"str":"Swinging over the world from below her airship was one of Amelia’s signature moves","name":"Amelia"},{"index":4247,"str":"Amelia’s producers had assured her the motor was fixed, and yet here she was again, hanging two hundred feet below the airship, and just above the trees","name":"Amelia"},{"index":4513,"str":"But Amelia was used to these kinds of situations; she wasn’t called Amelia Errhard for nothing; and she was in good contact with Frans","name":"Amelia"},{"index":4648,"str":"The wind was mild, and after some thought and discussion, Frans lowered the airship until Amelia could kick around in the uppermost leaves and twigs of the forest canopy, find one of the highest branches of an elm, and stand on it","name":"Amelia"},{"index":5028,"str":"“Now watch this, people,” Amelia said","name":"Amelia"},{"index":6183,"str":"Amelia ignored them all, Nicole in particular","name":"Amelia"},{"index":6631,"str":"Overlapping worlds, a stack of overlays, an accidental megastructure, a postcarbon landscape, each of the many networks performing its function in the great dance, and the habitat corridors providing a life space for their horizontal brothers and sisters, as Amelia called them on her broadcasts","name":"Amelia"},{"index":8989,"str":"Amelia narrated the sights to her audience with the astonishment common to all Manhattan tour guides","name":"Amelia"},{"index":10065,"str":"” Nicole sent a text saying it was time to wrap, so Amelia said, “Okay, folks, it’s been great having you along, thanks all of you for traveling with me","name":"Amelia"},{"index":11081,"str":"Amelia called in to confirm her arrival","name":"Amelia"},{"index":11529,"str":"She had considered living there, its blimp mast being so much higher, but the old tower had become fashionable, and even though Amelia was one of the most famous of the cloud stars, she couldn’t afford it","name":"Amelia"},{"index":12084,"str":"Ah yes: New York! Skyscrapers and everything! Amelia had been born and raised in Grants Pass, Oregon, and because of that she loved New York passionately, more than any of the natives ever knew enough to feel","name":"Amelia"},{"index":12864,"str":"Amelia had a teeny kitchen nook in her closet of an apartment, but like most residents of the Met she ate her dinners in the dining hall downstairs","name":"Amelia"},{"index":13209,"str":"It reminded Amelia of tadpoles in a pond","name":"Amelia"},{"index":13349,"str":"Vlade was at his table by the window overlooking the bacino, sitting with a woman Amelia didn’t know","name":"Amelia"},{"index":13451,"str":"Amelia approached, and Vlade introduced them: “Forty-twenty, this is Twenty-forty","name":"Amelia"},{"index":13537,"str":"Amelia Black, Inspector Gen Octaviasdottir","name":"Amelia"},{"index":13583,"str":"“Nice to meet you,” Amelia said as they shook hands","name":"Amelia"},{"index":13635,"str":"The policewoman said she had seen Amelia’s show","name":"Amelia"},{"index":13684,"str":"“Thanks,” Amelia said","name":"Amelia"},{"index":13957,"str":"The cooks rang the bell for last call, and Amelia stood to go see what was still there","name":"Amelia"},{"index":14215,"str":"As she dug in, Vlade and Gen talked about people Amelia didn’t know","name":"Amelia"},{"index":14431,"str":"“Well, I thought I was going to be here for a while,” Amelia said, “but this sounds too good to pass on","name":"Amelia"},{"index":15064,"str":"“Amelia, I was hoping you’d call! What do you think?”","name":"Amelia"},{"index":15118,"str":"“I think it’s crazy,” Amelia said","name":"Amelia"},{"index":15960,"str":"Amelia ended the call and looked up at Vlade and the policewoman","name":"Amelia"}]},{"chapter":12,"name":"","length":113,"links":[]},{"chapter":13,"name":"Charlotte","length":15182,"links":[{"index":5241,"str":"She had started walking to and from work after her excursion with Inspector Gen","name":"Gen"},{"index":5741,"str":"Before her walk with Inspector Gen, she had almost always taken the number four vaporetto to work and back, but the canals could be so jammed that often as she watched from a vapo she could see walkers on the boardwalks moving quite a bit faster than her","name":"Gen"},{"index":14328,"str":"When she got to her room, which was actually just a bed and desk in one of the dorm rooms, separated from the rest of her roommates by soundquilts, she found a message from Gen Octaviasdottir on her screen","name":"Gen"},{"index":14534,"str":"She tapped and Gen picked up","name":"Gen"},{"index":8960,"str":"Complaint about noise, priority in the boathouse, desire for a bigger freight elevator (Vlade rolling his eyes at this, mentioning size of elevator shaft, wondering if a taller elevator car would satisfy the complainer), dispute over the dues/work credit formula as applied to someone who thought cleaning the hallway on their floor was work deserving of a work credit","name":"Vlade"},{"index":10896,"str":"“Aeration,” Charlotte said in the way Vlade would say mildew","name":"Vlade"},{"index":11232,"str":"“One of the rich people who wants to buy in here is Amelia Black,” Vlade mentioned","name":"Vlade"},{"index":11963,"str":"Vlade thought the building’s thoughts, and she valued that","name":"Vlade"},{"index":12185,"str":"“I’ll tell her,” Vlade said","name":"Vlade"},{"index":12604,"str":"Vlade wanted his cathodic protection systems replaced on every steel beam in the building, and a new sewage processor to better capture and process their shit into fertilizer for the farm’s soil, and more say on the bacino’s aquaculture board","name":"Vlade"},{"index":13893,"str":"Vlade was scowling","name":"Vlade"},{"index":11232,"str":"“One of the rich people who wants to buy in here is Amelia Black,” Vlade mentioned","name":"Amelia"},{"index":0,"str":"Charlotte Armstrong’s alarm went off and she jabbed her wristpad","name":"Charlotte"},{"index":1926,"str":"Just as she was finishing with them and packing to go home, the mayor’s assistant, Tanganyika John, came in to ask if Charlotte could come over and help the mayor deal with an issue, great in importance yet vague in detail","name":"Charlotte"},{"index":2149,"str":"Charlotte was suspicious of this, as she was of John, a supercilious woman, slender and fashionable, whose only job was assisting the mayor, meaning she was one of the defensive ramparts that the mayor erected around herself as a matter of course","name":"Charlotte"},{"index":2619,"str":"Charlotte agreed with as much politeness as she could muster and followed John down the hall and up the elevator to the mayor’s administrative palace on the penthouse floor","name":"Charlotte"},{"index":2792,"str":"There three assistants just like John asked Charlotte to help the mayor write up a press release explaining why they had to impose immigration quotas for the good of the people already living in the city","name":"Charlotte"},{"index":2998,"str":"Charlotte immediately refused","name":"Charlotte"},{"index":3424,"str":"Charlotte was coming to believe that arrogance was a quality not just correlated with but a manifestation of stupidity, a result of stupidity","name":"Charlotte"},{"index":3567,"str":"In any case here Galina stood, vivid in the flesh, making the same request as if because it came from her Charlotte could not refuse, even though they had been enemies for almost ten years now","name":"Charlotte"},{"index":3910,"str":"In any case Charlotte quickly disabused her of the notion that a personal request carried any weight","name":"Charlotte"},{"index":4119,"str":"“Defending the borders isn’t possible when there are no borders,” Charlotte said","name":"Charlotte"},{"index":4323,"str":"Charlotte met it with a stony glare","name":"Charlotte"},{"index":4360,"str":"Through the pretended amusement and tolerance that followed, Charlotte saw the glint in the eye that indicated this was yet another little jab in their long battle, a parry-riposte that would be added to all the rest","name":"Charlotte"},{"index":4839,"str":"“This is New York,” Charlotte said","name":"Charlotte"},{"index":5078,"str":"Charlotte shrugged and excused herself","name":"Charlotte"},{"index":8691,"str":"Charlotte shrugged","name":"Charlotte"},{"index":10173,"str":"Marge’s contract with the co-op was rock solid, Charlotte knew this because she had helped write it, and so the Denver family was going to have to sell to the co-op for one hundred percent of Marge’s buy-in","name":"Charlotte"},{"index":10660,"str":"“Ten months?” Charlotte asked","name":"Charlotte"},{"index":10896,"str":"“Aeration,” Charlotte said in the way Vlade would say mildew","name":"Charlotte"},{"index":11503,"str":"“Would she work the co-op?” Charlotte asked, feeling skeptical","name":"Charlotte"},{"index":11946,"str":"Charlotte nodded","name":"Charlotte"},{"index":13318,"str":"“What?” Charlotte said, startled","name":"Charlotte"},{"index":13449,"str":"“But why?” Charlotte exclaimed","name":"Charlotte"},{"index":13785,"str":"“Fuck that,” Charlotte said","name":"Charlotte"},{"index":13928,"str":"“Let’s research it first,” Charlotte said","name":"Charlotte"},{"index":14074,"str":"Charlotte poured a stiff Irish coffee for herself, wanting both stimulation and sedation","name":"Charlotte"},{"index":14565,"str":"“Hi, it’s Charlotte","name":"Charlotte"}]},{"chapter":14,"name":"","length":45,"links":[]},{"chapter":15,"name":"Stefan and Roberto","length":17507,"links":[{"index":844,"str":"Stefan looked up- and downstream; nothing big coming either way","name":"Stefan"},{"index":909,"str":"Roberto pushed the throttle forward and their little prop under the stern lifted Stefan a few inches as they surged across the river","name":"Stefan"},{"index":1147,"str":"Roberto slowed while Stefan put on a long rubber glove","name":"Stefan"},{"index":2060,"str":"After a bit of tacking around and drifting, Stefan, consulting the GPS on their salvaged wristpad, announced they were over the spot they wanted","name":"Stefan"},{"index":2646,"str":"Roberto was their diver, because their wetsuit was too small for Stefan to get into","name":"Stefan"},{"index":3471,"str":"Stefan could just barely make him out","name":"Stefan"},{"index":3567,"str":"“All good?” Stefan inquired","name":"Stefan"},{"index":3824,"str":"Stefan let the nylon rope out hand over hand, allowing the bell to gently sink into the river with Roberto under it","name":"Stefan"},{"index":4020,"str":"When the rope went loose Stefan knew the bell was on the bottom, presumably next to or even on the cinder blocks marking their site","name":"Stefan"},{"index":4580,"str":"Stefan had clipped together two sets of her air hoses, making thirty feet of tubing, and Roberto was now seventeen feet under the surface, so all was well in that regard","name":"Stefan"},{"index":4752,"str":"Stefan couldn’t see much of Roberto, and even the bell was just a kind of glow in the dark water, lit by Roberto’s flashlight","name":"Stefan"},{"index":5110,"str":"Stefan tugged once on the oxygen tube","name":"Stefan"},{"index":5594,"str":"And indeed, down under the diving bell, Roberto turned on the detector, set it for gold, and jumped when the detector immediately started beeping—his head clonked against the side of the diving bell, and he shouted uselessly up to Stefan","name":"Stefan"},{"index":6648,"str":"Time passed for Stefan up in the boat","name":"Stefan"},{"index":7575,"str":"Stefan tugged on the oxygen tube three times","name":"Stefan"},{"index":7783,"str":"Stefan tugged back three times, harder than before","name":"Stefan"},{"index":8231,"str":"“Damn it, I’m pulling you,” Stefan announced loudly down at the bell","name":"Stefan"},{"index":8835,"str":"Stefan kept on cranking, knowing this was the only way to get Roberto to give up and surface","name":"Stefan"},{"index":9091,"str":"Pretty soon Stefan could see the top of the bell, and right after that Roberto burst onto the surface of the water, blowing out air, and then started in, not with curses but with triumphant whoops, “Yes! Yes!” followed by “I found it! We found it! The detector! It went off! We found it!” Then some violent hacking as he swallowed some river water","name":"Stefan"},{"index":9441,"str":"“Oh my God!” Quickly Stefan helped him over the rounded side of the boat, then hauled up the bell while Roberto started pulling himself out of the wetsuit","name":"Stefan"},{"index":10009,"str":"Freed of the wetsuit, standing in the wind in his wet shorts, he shut his eyes and Stefan sprayed him with a water bottle liberally dosed with bleach, and then Roberto toweled off his face","name":"Stefan"},{"index":10270,"str":"When Roberto was dried and dressed, he helped Stefan haul the diving bell onto the bow, and then they cast off from their underwater buoy and began to motor downstream, chattering all the while","name":"Stefan"},{"index":10466,"str":"“We’re going to run out of battery,” Stefan said","name":"Stefan"},{"index":11055,"str":"The various people around Madison Square whom they associated with were not fully aware of their situation, at least not formally, and they might not appreciate being asked for help if Stefan and Roberto were to name them as responsible parties","name":"Stefan"},{"index":11443,"str":"Stefan nodded","name":"Stefan"},{"index":11772,"str":"Stefan spotted a break in the river traffic, gunned their motor, and made as quick a crossing of the traffic lanes as he could, burning most of the rest of their battery’s juice","name":"Stefan"},{"index":12335,"str":"“So we found it,” Stefan said","name":"Stefan"},{"index":14126,"str":"“Together,” Stefan reminded him","name":"Stefan"},{"index":16016,"str":"“You swamped us,” Stefan shouted in turn, pointing down","name":"Stefan"},{"index":16609,"str":"“Hey don’t you live at the Met on Madison Square?” Stefan called suddenly","name":"Stefan"},{"index":909,"str":"Roberto pushed the throttle forward and their little prop under the stern lifted Stefan a few inches as they surged across the river","name":"Roberto"},{"index":1147,"str":"Roberto slowed while Stefan put on a long rubber glove","name":"Roberto"},{"index":1756,"str":"All looked good, and Roberto crawled under the edge of it to stick their new gear to Velcro strips on its inside wall","name":"Roberto"},{"index":2206,"str":"“Yes!” Roberto cried, and tossed one of their improvised underwater buoys overboard: two cinder blocks tied to a stolen nylon rope, the other end of rope tied to a buoy such that it would stay just under the surface even at low tide","name":"Roberto"},{"index":2646,"str":"Roberto was their diver, because their wetsuit was too small for Stefan to get into","name":"Roberto"},{"index":2854,"str":"When Roberto was all zipped in, gloved and face-masked, they lifted the cone over the side with its open end down, getting it onto the water as flatly as possible, so that as it dropped slowly into the turbid water, they saw that a good amount of air had been trapped under it","name":"Roberto"},{"index":3216,"str":"Roberto grabbed the end of the air hose and took their flashlight in the other hand, and with a deep breath he slipped over the side of the boat into the water","name":"Roberto"},{"index":3790,"str":"Roberto dove under the bell again","name":"Roberto"},{"index":3824,"str":"Stefan let the nylon rope out hand over hand, allowing the bell to gently sink into the river with Roberto under it","name":"Roberto"},{"index":4580,"str":"Stefan had clipped together two sets of her air hoses, making thirty feet of tubing, and Roberto was now seventeen feet under the surface, so all was well in that regard","name":"Roberto"},{"index":4752,"str":"Stefan couldn’t see much of Roberto, and even the bell was just a kind of glow in the dark water, lit by Roberto’s flashlight","name":"Roberto"},{"index":4878,"str":"But Roberto was now standing on an old asphalt surface of what once had been a parking lot, just behind the old riverfront in the south end of the Bronx","name":"Roberto"},{"index":5186,"str":"Down there Roberto would be deploying their metal detector, after detaching it from the inner wall of the bell","name":"Roberto"},{"index":5466,"str":"Roberto would use this detector to scan the ancient submerged asphalt and see if it detected anything under Mr","name":"Roberto"},{"index":5594,"str":"And indeed, down under the diving bell, Roberto turned on the detector, set it for gold, and jumped when the detector immediately started beeping—his head clonked against the side of the diving bell, and he shouted uselessly up to Stefan","name":"Roberto"},{"index":6224,"str":"Roberto’s heart rate was accelerating in time with the beeps, and he began to hyperventilate a little, muttering, “Oh my God, oh my God, oh my God","name":"Roberto"},{"index":7430,"str":"But for now it seemed to him that Roberto had been down there a long time","name":"Roberto"},{"index":7622,"str":"Down below, Roberto saw this but ignored it","name":"Roberto"},{"index":8161,"str":"Roberto could be hard to convince even when you were talking to him","name":"Roberto"},{"index":8319,"str":"They had a hand reel screwed to their plywood thwart, and now he looped the bell rope over the reel and began to turn hard on the crank, pulling the bell and therefore Roberto up from the bottom","name":"Roberto"},{"index":8516,"str":"Down below Roberto hurried to tack the paint can and metal detector against the inside of the bell before it rose over him","name":"Roberto"},{"index":8835,"str":"Stefan kept on cranking, knowing this was the only way to get Roberto to give up and surface","name":"Roberto"},{"index":9091,"str":"Pretty soon Stefan could see the top of the bell, and right after that Roberto burst onto the surface of the water, blowing out air, and then started in, not with curses but with triumphant whoops, “Yes! Yes!” followed by “I found it! We found it! The detector! It went off! We found it!” Then some violent hacking as he swallowed some river water","name":"Roberto"},{"index":9441,"str":"“Oh my God!” Quickly Stefan helped him over the rounded side of the boat, then hauled up the bell while Roberto started pulling himself out of the wetsuit","name":"Roberto"},{"index":9806,"str":"Roberto laughed","name":"Roberto"},{"index":10009,"str":"Freed of the wetsuit, standing in the wind in his wet shorts, he shut his eyes and Stefan sprayed him with a water bottle liberally dosed with bleach, and then Roberto toweled off his face","name":"Roberto"},{"index":10270,"str":"When Roberto was dried and dressed, he helped Stefan haul the diving bell onto the bow, and then they cast off from their underwater buoy and began to motor downstream, chattering all the while","name":"Roberto"},{"index":10614,"str":"“Whatever,” Roberto said","name":"Roberto"},{"index":10759,"str":"Roberto looked around the East River to check for traffic: crowded, as usual","name":"Roberto"},{"index":11055,"str":"The various people around Madison Square whom they associated with were not fully aware of their situation, at least not formally, and they might not appreciate being asked for help if Stefan and Roberto were to name them as responsible parties","name":"Roberto"},{"index":11457,"str":"He met Roberto’s eye and grinned","name":"Roberto"},{"index":14069,"str":"Roberto was a little faster under conditions like this","name":"Roberto"},{"index":15944,"str":"“Hey!” Roberto shouted furiously at the zoomer","name":"Roberto"},{"index":16289,"str":"“We ran out of battery power!” Roberto said","name":"Roberto"},{"index":16518,"str":"“Hey, give us a tow!” Roberto shouted furiously","name":"Roberto"},{"index":17109,"str":"“Thanks, mister,” Roberto said","name":"Roberto"},{"index":17289,"str":"“That’s why we’ll call it even,” Roberto said","name":"Roberto"}]},{"chapter":16,"name":"","length":879,"links":[]},{"chapter":17,"name":"Franklin","length":13435,"links":[{"index":3728,"str":"“What about you? Isn’t there anyone who shortens Franklin to Frank? I would think that would be a natural","name":"Franklin"},{"index":3969,"str":"And I liked Ben Franklin","name":"Franklin"},{"index":4047,"str":"“Not the Franklin saying I quote the most","name":"Franklin"}]},{"chapter":18,"name":"","length":118,"links":[]},{"chapter":19,"name":"Mutt and Jeff","length":4837,"links":[{"index":674,"str":"Mutt regards him","name":"Mutt"},{"index":1073,"str":"“Well …” Mutt ponders","name":"Mutt"},{"index":2020,"str":"Mutt nods","name":"Mutt"},{"index":2069,"str":"Mutt shakes his head, looks at Jeff","name":"Mutt"},{"index":2424,"str":"Mutt looks dubious","name":"Mutt"},{"index":2899,"str":"He walks over to a wall, runs his fingers over a tight seal in the shape of a door; there is no knob or keyhole inside this door-shaped line in the wall, although there is a rectangular line inside it, around waist height on Jeff, knee height on Mutt","name":"Mutt"},{"index":3430,"str":"Mutt stands, groans, looks around","name":"Mutt"},{"index":3960,"str":"Mutt comes back out of the bathroom and walks up and down the length of the room, placing his heels right against his toes and counting his steps, lips pulsing in and out as he calculates","name":"Mutt"},{"index":484,"str":"” Jeff shakes his head hard, then holds it in his hands","name":"Jeff"},{"index":1490,"str":"Jeff nods","name":"Jeff"},{"index":2069,"str":"Mutt shakes his head, looks at Jeff","name":"Jeff"},{"index":2215,"str":"Jeff’s eyes go round","name":"Jeff"},{"index":2382,"str":"Jeff nods","name":"Jeff"},{"index":2841,"str":"Jeff stands, holds his head with both hands","name":"Jeff"},{"index":2899,"str":"He walks over to a wall, runs his fingers over a tight seal in the shape of a door; there is no knob or keyhole inside this door-shaped line in the wall, although there is a rectangular line inside it, around waist height on Jeff, knee height on Mutt","name":"Jeff"},{"index":3268,"str":"” Jeff puts his ear to the wall","name":"Jeff"},{"index":4229,"str":"” He looks at Jeff","name":"Jeff"},{"index":4375,"str":"He puts his ear to the wall across from Jeff","name":"Jeff"}]},{"chapter":20,"name":"","length":587,"links":[]},{"chapter":21,"name":"Citizen","length":11229,"links":[]},{"chapter":22,"name":"","length":120,"links":[]},{"chapter":23,"name":"Inspector Gen","length":13581,"links":[{"index":2605,"str":"Ralph Muttchopf did his graduate degree in computer science","name":"Mutt"},{"index":5357,"str":"Then more recently, Rosen and Muttchopf did some contract work for Vinson’s hedge fund, Alban Albany, enough to get them tax forms for last year","name":"Mutt"},{"index":5651,"str":"Vinson has any number of colleagues and acquaintances, and so did Muttchopf and Rosen","name":"Mutt"},{"index":2666,"str":"Jeffrey Rosen had a degree in philosophy, and he worked as a congressional staffer for the Senate Finance Committee about fifteen years ago","name":"Jeff"},{"index":4191,"str":"“Anyway, he doesn’t talk about family much, so I just thought this Jeff Rosen might turn out to be one of his cousins","name":"Jeff"},{"index":4403,"str":"But go on—it was Vinson who Jeffrey Rosen is related to, you say","name":"Jeff"},{"index":0,"str":"Gen Octaviasdottir usually woke at sunrise","name":"Gen"},{"index":670,"str":"A photo of a statue of a pharaoh and his sister/wife, which Gen’s father had thought looked like him and Gen’s mom","name":"Gen"},{"index":807,"str":"Gen kept meaning to take all these down, they were dusty, but she never got around to it","name":"Gen"},{"index":896,"str":"Her parents had had a good marriage, but Gen’s one youthful attempt had failed badly, and after that she had let the NYPD occupy her time","name":"Gen"},{"index":2122,"str":"Gen nodded and pulled out her pad","name":"Gen"},{"index":3714,"str":"“Good idea,” Gen said, and was relieved when Charlotte laughed","name":"Gen"},{"index":4155,"str":"“Wow,” Gen said","name":"Gen"},{"index":4519,"str":"“It’s just a way in,” Gen said","name":"Gen"},{"index":5869,"str":"Gen watched her closely as she said, “Please don’t say anything about this to Larry Jackman","name":"Gen"},{"index":6870,"str":"Gen shrugged","name":"Gen"},{"index":6946,"str":"Gen thought it over","name":"Gen"},{"index":7899,"str":"Gen shrugged","name":"Gen"},{"index":8107,"str":"“I probably should,” Gen allowed","name":"Gen"},{"index":8461,"str":"Gen had a tendency to see patterns","name":"Gen"},{"index":9033,"str":"“I hope it doesn’t happen,” Gen said","name":"Gen"},{"index":9238,"str":"Gen said, “How can you tell if their bid is twice what the building’s worth? How can anyone tell what anything is worth these days?”","name":"Gen"},{"index":9838,"str":"“The greatest generation,” Gen quoted","name":"Gen"},{"index":10142,"str":"Gen nodded","name":"Gen"},{"index":10695,"str":"Gen nodded as she regarded Charlotte","name":"Gen"},{"index":10938,"str":"Gen was beginning to see why her youthful marriage might have failed: a financier and a social worker walk into a bar …","name":"Gen"},{"index":11059,"str":"But in fact Gen saw no signs of dissembling","name":"Gen"},{"index":11454,"str":"“I think that will be the majority opinion,” Gen said reassuringly","name":"Gen"},{"index":12046,"str":"Charlotte was shaking her head through most of what Gen had said","name":"Gen"},{"index":13055,"str":"And Gen could tell she had managed to irritate Charlotte more than once","name":"Gen"},{"index":6564,"str":"“They asked Vlade, and Vlade asked me, and I met with them and thought they were okay, so I asked the residency board to let them stay on a temp permit","name":"Vlade"},{"index":12111,"str":"“Vlade is this building","name":"Vlade"},{"index":12458,"str":"No, Vlade loves this place, I know that","name":"Vlade"},{"index":1221,"str":"Down to the dining room for breakfast with Charlotte Armstrong","name":"Charlotte"},{"index":1735,"str":"She got to the dining hall at the same time Charlotte did, both right on time","name":"Charlotte"},{"index":1922,"str":"Charlotte took her coffee white","name":"Charlotte"},{"index":1995,"str":"“So did your assistant find out anything about our missing guys?” Charlotte asked after they sat down","name":"Charlotte"},{"index":3465,"str":"Charlotte rolled her eyes","name":"Charlotte"},{"index":3573,"str":"” Charlotte shrugged","name":"Charlotte"},{"index":3714,"str":"“Good idea,” Gen said, and was relieved when Charlotte laughed","name":"Charlotte"},{"index":3778,"str":"“Yes,” Charlotte admitted, “always a good idea","name":"Charlotte"},{"index":4172,"str":"Charlotte shrugged","name":"Charlotte"},{"index":4962,"str":"“And Jackman now runs the Fed,” Charlotte added, looking a little grim","name":"Charlotte"},{"index":8141,"str":"Charlotte nodded at this, but then something else struck her: “Everyone’s going to know pretty soon about something that came up at the last board meeting","name":"Charlotte"},{"index":9172,"str":"Charlotte shrugged","name":"Charlotte"},{"index":9371,"str":"“Comparisons to similar deals,” Charlotte said","name":"Charlotte"},{"index":10695,"str":"Gen nodded as she regarded Charlotte","name":"Charlotte"},{"index":10732,"str":"She was on the watch for any signs of dissembling, because Charlotte had connections with the missing men in more ways than one, so there was reason to be attentive","name":"Charlotte"},{"index":11103,"str":"On the contrary, Charlotte seemed very open and frank","name":"Charlotte"},{"index":12046,"str":"Charlotte was shaking her head through most of what Gen had said","name":"Charlotte"},{"index":12809,"str":"“Yes,” Charlotte replied heavily","name":"Charlotte"},{"index":13055,"str":"And Gen could tell she had managed to irritate Charlotte more than once","name":"Charlotte"},{"index":13525,"str":"“Thanks,” Charlotte said","name":"Charlotte"}]},{"chapter":24,"name":"","length":397,"links":[]},{"chapter":25,"name":"Vlade","length":13452,"links":[{"index":0,"str":"Mayday,” the Met said from Vlade’s wall monitor","name":"Vlade"},{"index":487,"str":"Vlade clumped downstairs to the sub-basement and the lights came on ahead of him as he moved","name":"Vlade"},{"index":2236,"str":"It was actually a graphenated composite, but as it was transparent and shiny, Vlade like everyone else called it diamond","name":"Vlade"},{"index":4602,"str":"“Vlade,” the Met said in his earbud, “mayday","name":"Vlade"},{"index":5318,"str":"Right now it was high tide, so there would be a little more pressure on any submarine leaks, but for two leaks to spring at almost the same time struck Vlade as extremely suspicious, especially given the corner position and drilled look of the one below","name":"Vlade"},{"index":7050,"str":"“I’m going to take a quick dive,” Vlade told him, which caused Su to frown","name":"Vlade"},{"index":7126,"str":"Dives were never supposed to be solo, but Vlade did it all the time around the building, accompanied only by a little sub sled","name":"Vlade"},{"index":7255,"str":"“I’ll keep the phone on,” Su said, to remind him, and Vlade nodded and began the somewhat arduous process of getting his wetsuit on","name":"Vlade"},{"index":9567,"str":"Vlade had spent ten years in the city’s water division, working on sewage lines, utilidors, subway tunnels, and aquafarms, mostly","name":"Vlade"},{"index":11387,"str":"It wasn’t clear to Vlade that this would be particularly useful","name":"Vlade"},{"index":11452,"str":"Closing the barn door after the horses had leaked, as far as Vlade and many other water rats were concerned, but the hydrologists had declared it would help the situation, and so slowly it got done","name":"Vlade"},{"index":11719,"str":"Looking at the edge of the sealant and sheeting and the beginning of bare street concrete, now a canal bottom, Vlade could feel in his gut why the hydrologists had wanted to try something","name":"Vlade"},{"index":13184,"str":"He called up Charlotte Armstrong","name":"Charlotte"},{"index":13217,"str":"“Charlotte, where are you?”","name":"Charlotte"}]},{"chapter":26,"name":"","length":435,"links":[]},{"chapter":27,"name":"Amelia","length":14714,"links":[{"index":0,"str":"Amelia’s airship the Assisted Migration was a Friedrichshafen Deluxe Midi, and she loved it","name":"Amelia"},{"index":2069,"str":"And like millions of other aircraft occupants, for many years Amelia had therefore not gone down","name":"Amelia"},{"index":5139,"str":"After Frans attached to an airship mast at the edge of town and was pulled to the ground by a local crew, Amelia got out and greeted a clutch of locals","name":"Amelia"},{"index":5369,"str":"Amelia shook everyone’s hand and thanked them for hosting her, filming all the while with a swarm of camera flies","name":"Amelia"},{"index":5547,"str":"“We’re approaching the polar bear jail in Churchill,” Amelia voice-overed unnecessarily as she filmed","name":"Amelia"},{"index":6724,"str":"Amelia’s producers had prepared the room in advance of departure and stocked the craft’s freezers and refrigerators with the seal steaks needed to feed them en route","name":"Amelia"},{"index":6892,"str":"As the local program officers used a crane to hoist the drugged and netted bears into the pickup truck, then drove them over to the airship, Amelia filmed and spoke her voice-over, ad-libbing in the knowledge that later editing would change it all anyway","name":"Amelia"},{"index":8560,"str":"A week later they ran into a tropical storm coming north from Trinidad and Tobago, and Amelia asked Frans to head for the west fringe of the storm’s circulation, which would give her viewers a dramatic edge-on view of what might become a hurricane, while also pushing them southward in its counterclockwise flow","name":"Amelia"},{"index":8872,"str":"The storm was now named Harold, which was Amelia’s younger brother’s name, so she started calling the storm Little Brother","name":"Amelia"},{"index":9182,"str":"“That gives us a net assist south of about a hundred and eighty kilometers per hour,” Amelia informed her future audience, “which is great, even if it only lasts for a few hours","name":"Amelia"},{"index":9862,"str":"But it sure didn’t sound like it to Amelia","name":"Amelia"},{"index":10268,"str":"They were buffeted hard, and whether the bears were still complaining was hard to tell, as it was too loud to hear anything, but Amelia’s stomach was still vibrating like a drumhead next to another drum getting hammered, so it seemed like they probably were","name":"Amelia"},{"index":10527,"str":"“Hold on, folks!” Amelia said loudly","name":"Amelia"},{"index":11248,"str":"There was turbulence as the whirlpool of wind interacted with the slower air around it; things would go better when they moved a bit farther into the hurricane, as Amelia explained, not for the first time","name":"Amelia"},{"index":11906,"str":"“This bumpiness is part of the laminar flow,” Amelia narrated","name":"Amelia"},{"index":12069,"str":"Amelia felt sure it was not usually this bumpy inside clouds, even hurricane clouds","name":"Amelia"},{"index":12414,"str":"“I don’t know,” Amelia announced, “it doesn’t make sense, but I’m wondering if this rocking is being caused by the bears?”","name":"Amelia"},{"index":13620,"str":"Amelia went to the tool closet in the hall, opened it, and took a tranquilizer dart gun from the mounting on the back wall","name":"Amelia"}]},{"chapter":28,"name":"","length":831,"links":[]},{"chapter":29,"name":"Stefan and Roberto","length":21967,"links":[{"index":17001,"str":"The boys had tended to scoff at the bad stories told about such collapses, but now they were remembering how Vlade always called the intertidal the death zone","name":"Vlade"},{"index":0,"str":"Stefan and Roberto had not found a chance to recharge the battery that powered their boat, so they walked on skybridges west and got on the Sixth vapo north to go see their friend Mr","name":"Stefan"},{"index":1882,"str":"People like Stefan and Roberto, who loved to join the skimboarders who congregated when the rising tide, coming up both Broadway and Sixth, combined to surge hard up the slight incline of Sixth, each advance of the white foam hissing north with startling rapidity, especially if pushed by a south wind","name":"Stefan"},{"index":2603,"str":"All that action meant that if you were riding the surge on a skimboard, as Stefan and Roberto soon were, you could cut around on the mini-breaks, shoot across the street from curb to curb, turn on a dime in the curbslush, or jump the curb and turn in doorways, sometimes even catching the rebound wave coming off buildings and jumping off the curb back into the street","name":"Stefan"},{"index":2973,"str":"Stefan and Roberto joined the group with some whoops to announce their presence","name":"Stefan"},{"index":9361,"str":"Stefan and Roberto didn’t like to talk about this, but neither of them could read","name":"Stefan"},{"index":9643,"str":"“We think we did,” Stefan said","name":"Stefan"},{"index":10043,"str":"“But the thing is, how deep could it be down?” Stefan asked","name":"Stefan"},{"index":15116,"str":"“More like twenty,” Stefan said","name":"Stefan"},{"index":15317,"str":"“But the metal detector detected it,” Stefan pointed out","name":"Stefan"},{"index":15475,"str":"Stefan wasn’t so sure","name":"Stefan"},{"index":15761,"str":"Stefan nodded uncertainly","name":"Stefan"},{"index":16276,"str":"“Look at the walls!” Stefan said, shocked","name":"Stefan"},{"index":19365,"str":"Stefan rolled his eyes and suggested they get going somewhere","name":"Stefan"},{"index":19428,"str":"Then Hexter’s own building groaned immensely behind them, and Stefan and Roberto picked up the old man by the elbows and moved him as fast as they could over the wreckage in the canal","name":"Stefan"},{"index":20404,"str":"“You okay?” Stefan asked him again","name":"Stefan"},{"index":20628,"str":"Stefan said to Roberto, “We’re taking him to the Met?”","name":"Stefan"},{"index":0,"str":"Stefan and Roberto had not found a chance to recharge the battery that powered their boat, so they walked on skybridges west and got on the Sixth vapo north to go see their friend Mr","name":"Roberto"},{"index":1882,"str":"People like Stefan and Roberto, who loved to join the skimboarders who congregated when the rising tide, coming up both Broadway and Sixth, combined to surge hard up the slight incline of Sixth, each advance of the white foam hissing north with startling rapidity, especially if pushed by a south wind","name":"Roberto"},{"index":2603,"str":"All that action meant that if you were riding the surge on a skimboard, as Stefan and Roberto soon were, you could cut around on the mini-breaks, shoot across the street from curb to curb, turn on a dime in the curbslush, or jump the curb and turn in doorways, sometimes even catching the rebound wave coming off buildings and jumping off the curb back into the street","name":"Roberto"},{"index":2973,"str":"Stefan and Roberto joined the group with some whoops to announce their presence","name":"Roberto"},{"index":7860,"str":"“Essence of New York,” Roberto noted as they shuffled down the dark hallway to the end door","name":"Roberto"},{"index":9361,"str":"Stefan and Roberto didn’t like to talk about this, but neither of them could read","name":"Roberto"},{"index":9560,"str":"“We found it,” Roberto said","name":"Roberto"},{"index":9893,"str":"“It was pinging like crazy,” Roberto said","name":"Roberto"},{"index":12417,"str":"“So that’s good, right?” Roberto asks","name":"Roberto"},{"index":14232,"str":"“But how deep?” Roberto prompted, after Hexter seemed to be taking a little nap","name":"Roberto"},{"index":15438,"str":"“So we can do it,” Roberto declared","name":"Roberto"},{"index":15647,"str":"“We’ll have to circle the hole, move the dirt off in different directions,” Roberto said","name":"Roberto"},{"index":16482,"str":"“We gotta get out of here!” Roberto told Mr","name":"Roberto"},{"index":16648,"str":"The two boys crouched and threw books around carefully but swiftly until Roberto came upon the glasses; they were still intact","name":"Roberto"},{"index":19315,"str":"“Really?” Roberto said","name":"Roberto"},{"index":19428,"str":"Then Hexter’s own building groaned immensely behind them, and Stefan and Roberto picked up the old man by the elbows and moved him as fast as they could over the wreckage in the canal","name":"Roberto"},{"index":20628,"str":"Stefan said to Roberto, “We’re taking him to the Met?”","name":"Roberto"},{"index":20777,"str":"“Come on,” Roberto said, “we’ll be fine","name":"Roberto"},{"index":20867,"str":"“No,” Roberto said","name":"Roberto"}]},{"chapter":30,"name":"","length":331,"links":[]},{"chapter":31,"name":"Franklin","length":39962,"links":[{"index":38305,"str":"Gen told us we could use it again","name":"Gen"},{"index":25750,"str":"“Franklin Garr,” I said","name":"Franklin"},{"index":24519,"str":"“Back at the Met, maybe? Vlade will know what to do","name":"Vlade"},{"index":25298,"str":"She nodded and held out her arm, and he tapped on it and then said, “Vlade, our pad got soaked, but can you let us dry off in your office maybe? We have a friend whose building got knocked over","name":"Vlade"},{"index":33652,"str":"We reconvened at a single table: the boys and their ancient friend; Jojo and Charlotte the chairperson; the super, whose name was Vlade, very apropos, Vlade the Impaler, face like a Ukrainian executioner; and me","name":"Vlade"},{"index":36296,"str":"I wondered what that could be about; possibly they didn’t want Vlade thinking about them going back to the intertidal","name":"Vlade"},{"index":36415,"str":"Indeed the super was frowning, but the taller boy saw this and said, “Come on, Vlade, we’re there every day","name":"Vlade"},{"index":38200,"str":"“Isn’t that a crime scene?” Vlade asked","name":"Vlade"},{"index":39193,"str":"Vlade nodded at them and joined us as we headed toward the elevators, leaving the two women behind","name":"Vlade"},{"index":27610,"str":"Charlotte’s the one to talk to about that","name":"Charlotte"},{"index":27895,"str":"And I’ll see if Heloise can rustle up a place for him, if Charlotte says it’s okay","name":"Charlotte"},{"index":30291,"str":"And lo and behold here she was herself, in conversation with the boys and the old man: Charlotte Armstrong, looking frazzled and intense, vivid and imposing","name":"Charlotte"},{"index":31189,"str":"Chairwoman Charlotte and Jojo followed me to the food windows in the dining hall, and I flashed my meat card to the clerk while listening to the two women talking","name":"Charlotte"},{"index":33652,"str":"We reconvened at a single table: the boys and their ancient friend; Jojo and Charlotte the chairperson; the super, whose name was Vlade, very apropos, Vlade the Impaler, face like a Ukrainian executioner; and me","name":"Charlotte"},{"index":34209,"str":"Still, everyone started by asking the old man how he was feeling, and Charlotte, hearing his story and squinting unhappily as she no doubt contemplated our building’s nonexistent or even negative vacancy rate, offered him a temporary place to stay, “until you can get back into your place or find something more suitable","name":"Charlotte"},{"index":34587,"str":"Charlotte said, “We’re full right now, that’s the problem","name":"Charlotte"},{"index":35164,"str":"Jojo didn’t seem to notice this, but Charlotte gave me a grateful look as I rose","name":"Charlotte"},{"index":36680,"str":"Meanwhile Charlotte and Jojo were getting acquainted","name":"Charlotte"},{"index":36769,"str":"Charlotte frowned","name":"Charlotte"},{"index":36976,"str":"Charlotte did not look impressed","name":"Charlotte"},{"index":37446,"str":"“Indeed,” Charlotte said, squinting slightly","name":"Charlotte"},{"index":37779,"str":"Charlotte gave me a dirty look, but Jojo’s was even worse","name":"Charlotte"},{"index":37956,"str":"“We haven’t worked out where that is yet,” Charlotte said","name":"Charlotte"},{"index":38241,"str":"Charlotte shook her head","name":"Charlotte"},{"index":38443,"str":"“Okay then,” Charlotte said","name":"Charlotte"},{"index":38628,"str":"Charlotte seemed unaware of their unease","name":"Charlotte"},{"index":39669,"str":"“I’ll be up in a bit,” Charlotte said","name":"Charlotte"},{"index":27203,"str":"“Roberto and Stefan, spreaders of chaos","name":"Stefan"},{"index":34876,"str":"Either Roberto or Stefan","name":"Stefan"},{"index":34980,"str":"The taller of the two boys, this was probably Stefan, said, “It’s tilted like diagonal","name":"Stefan"},{"index":27203,"str":"“Roberto and Stefan, spreaders of chaos","name":"Roberto"},{"index":34844,"str":"He was Roberto, I was learning","name":"Roberto"},{"index":34876,"str":"Either Roberto or Stefan","name":"Roberto"},{"index":36175,"str":"“We’ll go tomorrow,” Roberto told him, with a little headshake to his ancient friend that said Don’t talk about this now","name":"Roberto"}]},{"chapter":32,"name":"","length":1248,"links":[]},{"chapter":33,"name":"Citizen","length":17422,"links":[]},{"chapter":34,"name":"","length":358,"links":[]},{"chapter":35,"name":"Mutt and Jeff","length":10677,"links":[{"index":2266,"str":"Mutt shakes his head","name":"Mutt"},{"index":3879,"str":"Jeff stares at Mutt for a while","name":"Mutt"},{"index":4082,"str":"“Like Gaia, Mutt","name":"Mutt"},{"index":4713,"str":"“Mutt, please","name":"Mutt"},{"index":4858,"str":"Mutt shrugs","name":"Mutt"},{"index":5160,"str":"Mutt stares at him","name":"Mutt"},{"index":5439,"str":"Mutt finds himself standing up, looking at Jeff, who is regarding the floor","name":"Mutt"},{"index":6618,"str":"Mutt stares at him","name":"Mutt"},{"index":7040,"str":"Mutt throws up his hands","name":"Mutt"},{"index":7330,"str":"Mutt goes and sits down on his bed","name":"Mutt"},{"index":7834,"str":"Mutt shakes his head","name":"Mutt"},{"index":9204,"str":"“We have those here,” Mutt points out","name":"Mutt"},{"index":9269,"str":"“It’s not all we need,” Mutt persists","name":"Mutt"},{"index":10049,"str":"Mutt shakes his head","name":"Mutt"},{"index":182,"str":"“Jeff, stop it","name":"Jeff"},{"index":803,"str":"Jeff shook his head","name":"Jeff"},{"index":1572,"str":"Jeff sighs","name":"Jeff"},{"index":3879,"str":"Jeff stares at Mutt for a while","name":"Jeff"},{"index":4921,"str":"Jeff can only nod to this","name":"Jeff"},{"index":4995,"str":"Jeff dismisses it with the back of his hand: “I was gonna introduce a meta-tap, where every transaction made over the CME sent a point to the SEC’s operating fund","name":"Jeff"},{"index":5439,"str":"Mutt finds himself standing up, looking at Jeff, who is regarding the floor","name":"Jeff"},{"index":5560,"str":"Jeff shrugged","name":"Jeff"},{"index":7952,"str":"“But just their money,” Jeff says","name":"Jeff"},{"index":8948,"str":"Jeff sighs","name":"Jeff"},{"index":9243,"str":"Jeff heaves another sigh","name":"Jeff"},{"index":9470,"str":"Jeff nods","name":"Jeff"},{"index":9934,"str":"“I tried,” Jeff says","name":"Jeff"},{"index":10217,"str":"Jeff frowns","name":"Jeff"},{"index":10605,"str":"“Nah,” Jeff says after a while","name":"Jeff"}]},{"chapter":36,"name":"","length":394,"links":[]},{"chapter":37,"name":"Charlotte","length":19229,"links":[{"index":142,"str":"Going out with Franklin Garr, and like him working in finance, meaning Charlotte didn’t exactly know what","name":"Franklin"},{"index":684,"str":"But she liked Franklin Garr, strange but true, so maybe that would extend to this woman","name":"Franklin"},{"index":5813,"str":"And check in with Franklin again","name":"Franklin"},{"index":5978,"str":"Vlade and Franklin and the boys and their elderly friend were all gathered outside the hotello, seated on chairs and little gardening stools","name":"Franklin"},{"index":6779,"str":"“It makes it dangerous,” Franklin opined","name":"Franklin"},{"index":6910,"str":"“Franklin Garr here","name":"Franklin"},{"index":7124,"str":"“Sorry!” Franklin said","name":"Franklin"},{"index":7465,"str":"Vlade glanced at Franklin with a surprised look","name":"Franklin"},{"index":7654,"str":"“Just the same,” Franklin said confidently","name":"Franklin"},{"index":8492,"str":"“Yikes,” Franklin said after a wondering silence","name":"Franklin"},{"index":8877,"str":"The hotello was really just a walk-in tent, so Charlotte and Franklin and Jojo stayed outside it as Vlade led the old man in with the two boys","name":"Franklin"},{"index":10499,"str":"The boys sat at his feet, Vlade on the chair next to Charlotte, Franklin and Jojo on the chairs beyond them","name":"Franklin"},{"index":15294,"str":"“What else would they wear?” Franklin asked, and Jojo elbowed him so hard his chair squeaked, and he did too","name":"Franklin"},{"index":4290,"str":"“Excuse me Miz Armstrong, it’s Amelia Black, I live in the Met when I’m in New York? I was trying to reach Vlade but I couldn’t get him","name":"Vlade"},{"index":5551,"str":"Well, wait just a second, I’m almost to the farm and Vlade is there","name":"Vlade"},{"index":5978,"str":"Vlade and Franklin and the boys and their elderly friend were all gathered outside the hotello, seated on chairs and little gardening stools","name":"Vlade"},{"index":6121,"str":"Charlotte interrupted them: “Vlade, can you help us a second here? I’ve got Amelia on the phone, and she’s in a situation on her blimp there, the polar bears have gotten loose","name":"Vlade"},{"index":6299,"str":"That got their attention instantly, and Vlade said loudly, “Amelia, is that true? Are you there?”","name":"Vlade"},{"index":6594,"str":"Vlade shook his head as he listened","name":"Vlade"},{"index":7465,"str":"Vlade glanced at Franklin with a surprised look","name":"Vlade"},{"index":7832,"str":"Again Vlade agreed this was a good idea","name":"Vlade"},{"index":8097,"str":"“There’ll be a bucket in most tool closets,” Vlade said","name":"Vlade"},{"index":8762,"str":"“No, we were just going to do that,” Vlade said","name":"Vlade"},{"index":8877,"str":"The hotello was really just a walk-in tent, so Charlotte and Franklin and Jojo stayed outside it as Vlade led the old man in with the two boys","name":"Vlade"},{"index":10499,"str":"The boys sat at his feet, Vlade on the chair next to Charlotte, Franklin and Jojo on the chairs beyond them","name":"Vlade"},{"index":19074,"str":"And I think Vlade and I should get back to Amelia, see how she’s doing","name":"Vlade"},{"index":4290,"str":"“Excuse me Miz Armstrong, it’s Amelia Black, I live in the Met when I’m in New York? I was trying to reach Vlade but I couldn’t get him","name":"Amelia"},{"index":4772,"str":"“Well,” Amelia said, “basically my polar bears have taken over my airship","name":"Amelia"},{"index":6121,"str":"Charlotte interrupted them: “Vlade, can you help us a second here? I’ve got Amelia on the phone, and she’s in a situation on her blimp there, the polar bears have gotten loose","name":"Amelia"},{"index":6299,"str":"That got their attention instantly, and Vlade said loudly, “Amelia, is that true? Are you there?”","name":"Amelia"},{"index":6397,"str":"“Yes,” Amelia said unhappily","name":"Amelia"},{"index":6452,"str":"Amelia described the sequence of questionable moves that had gotten her locked in a closet on an airship filled with polar bears on the loose","name":"Amelia"},{"index":6632,"str":"“Well, Amelia,” he said when she finished","name":"Amelia"},{"index":6854,"str":"“I can hear that,” Amelia reminded them","name":"Amelia"},{"index":7873,"str":"“Okay,” Amelia said","name":"Amelia"},{"index":8338,"str":"“Amelia?” Charlotte asked","name":"Amelia"},{"index":19074,"str":"And I think Vlade and I should get back to Amelia, see how she’s doing","name":"Amelia"},{"index":0,"str":"Charlotte looked carefully at the woman Jojo as they sat across from each other at the long dining hall table","name":"Charlotte"},{"index":142,"str":"Going out with Franklin Garr, and like him working in finance, meaning Charlotte didn’t exactly know what","name":"Charlotte"},{"index":328,"str":"Charlotte didn’t like her","name":"Charlotte"},{"index":1872,"str":"Charlotte stared at the woman","name":"Charlotte"},{"index":2471,"str":"Aeration was a term used all the time on the left side of the cloud where Charlotte tended to read commentary, but obviously this woman didn’t read there","name":"Charlotte"},{"index":2914,"str":"Charlotte stared at her","name":"Charlotte"},{"index":2979,"str":"Jojo shrugged gracefully, and seeing that Charlotte began to hate her in earnest","name":"Charlotte"},{"index":3180,"str":"Charlotte didn’t like it when people pretended to believe things you knew they couldn’t really believe; it was just a brush-off, an arrogance shading toward contempt","name":"Charlotte"},{"index":3347,"str":"By this gesture she was saying Charlotte wasn’t worth talking to","name":"Charlotte"},{"index":3414,"str":"Charlotte shrugged back, a crude mirroring","name":"Charlotte"},{"index":3763,"str":"Charlotte shook her head","name":"Charlotte"},{"index":4152,"str":"Charlotte was about to say something sharp when her pad played the first bars of Tchaikovsky’s Fourth Symphony","name":"Charlotte"},{"index":4263,"str":"Charlotte tapped the pad","name":"Charlotte"},{"index":4682,"str":"“What?” Charlotte began walking toward the elevator, and for some reason Jojo came along","name":"Charlotte"},{"index":5671,"str":"Jojo raised her eyebrows when Charlotte looked at her, and said in a low voice, “Sorry, I just want to hear what happens here, if that’s okay","name":"Charlotte"},{"index":5849,"str":"“Fine by me,” Charlotte said","name":"Charlotte"},{"index":6121,"str":"Charlotte interrupted them: “Vlade, can you help us a second here? I’ve got Amelia on the phone, and she’s in a situation on her blimp there, the polar bears have gotten loose","name":"Charlotte"},{"index":7147,"str":"With an uneasy glance at Jojo, now standing beside him (which had pleased him greatly, Charlotte saw), he added, “Are you in touch with the autopilot? Can you fly the thing?”","name":"Charlotte"},{"index":7943,"str":"“Wouldn’t miss it for the world, dear,” Charlotte said","name":"Charlotte"},{"index":8338,"str":"“Amelia?” Charlotte asked","name":"Charlotte"},{"index":8541,"str":"Charlotte saw Jojo elbow him in the ribs, saw him wince and then ignore it, eyes slightly crossed","name":"Charlotte"},{"index":8687,"str":"Charlotte gestured at the hotello door","name":"Charlotte"},{"index":8877,"str":"The hotello was really just a walk-in tent, so Charlotte and Franklin and Jojo stayed outside it as Vlade led the old man in with the two boys","name":"Charlotte"},{"index":9020,"str":"To Charlotte this viewing was a formality only; beggars can’t be choosers","name":"Charlotte"},{"index":10499,"str":"The boys sat at his feet, Vlade on the chair next to Charlotte, Franklin and Jojo on the chairs beyond them","name":"Charlotte"},{"index":10632,"str":"“I hate those chopsticks,” Charlotte said to the old man, gesturing at the two glass splinters","name":"Charlotte"},{"index":11398,"str":"“Sure,” Charlotte said","name":"Charlotte"},{"index":15403,"str":"Charlotte began to like Jojo a little better","name":"Charlotte"},{"index":18587,"str":"“Some of us could help too,” Charlotte said","name":"Charlotte"},{"index":18689,"str":"They’re plotting something, Charlotte thought","name":"Charlotte"},{"index":15814,"str":"“What did you do when you got off the berm?” Stefan asked","name":"Stefan"},{"index":17509,"str":"“Hard to believe,” Stefan said again","name":"Stefan"},{"index":17996,"str":"“Didn’t they all get flooded too?” Stefan asked","name":"Stefan"},{"index":11678,"str":"“You saw it?” Roberto asked incredulously, looking up at his face","name":"Roberto"},{"index":11948,"str":"“Nope,” Roberto said","name":"Roberto"},{"index":13479,"str":"“I’d like to see that,” Roberto said","name":"Roberto"},{"index":13614,"str":"“You were there?” Roberto asked","name":"Roberto"},{"index":14666,"str":"“So you got off the berm?” Roberto said","name":"Roberto"},{"index":15069,"str":"“Weren’t there people?” Roberto asked","name":"Roberto"},{"index":17098,"str":"“So was your home flooded?” Roberto said","name":"Roberto"},{"index":18171,"str":"“Do they know how many died that day?” Roberto asked","name":"Roberto"}]},{"chapter":38,"name":"","length":71,"links":[]},{"chapter":39,"name":"Amelia","length":13474,"links":[{"index":0,"str":"Frans tilted Amelia’s airship so far toward the vertical, bow high, stern low, that Amelia was forced to sit on the back wall of her closet, in a clutter of stuff","name":"Amelia"},{"index":9815,"str":"“Oh!” Amelia said to it","name":"Amelia"},{"index":10555,"str":"So Amelia without the slightest reluctance shot the bear in the shoulder, then again in the chest; then she landed on the back wall right next to it","name":"Amelia"},{"index":10756,"str":"It brushed it off, then growled loudly, so loudly that Amelia instinctively jumped up again and got another surprising helium assist, afterward flailing a bit as she pendulumed around the air of the room right above the bear, who waved at her woozily","name":"Amelia"},{"index":11008,"str":"Then the bear grew content to lie down and sleep it off, and Amelia avoided plunging through the open door to the hall by way of some deft footwork, after which she landed and sat there on the back wall beside the open door, now like a trapdoor to doom, hyperventilating","name":"Amelia"},{"index":11296,"str":"When the bear seemed to be really out, Amelia asked Frans to right the ship","name":"Amelia"},{"index":11756,"str":"Even a dog could do that with Amelia, and this bear weighed about seven hundred pounds","name":"Amelia"},{"index":11844,"str":"“If I had a lever, I could move the bear,” Amelia said aloud","name":"Amelia"},{"index":12184,"str":"Amelia had to think out the directions, then tell Frans which way to tilt","name":"Amelia"},{"index":12477,"str":"When it was close to the edge, Amelia used a broom as a crowbar and levered the bear into the doorway","name":"Amelia"},{"index":12580,"str":"Prepared for this moment, Amelia ordered Frans to shift more off the vertical at the same moment the bear rolled into the hole, and it seemed like Frans tilted fast enough that when the bear hit the stern end of the hall, it was more sliding than falling","name":"Amelia"},{"index":12910,"str":"“Now I have to close the door!” Amelia cried, and she jumped through the doorway still holding the bags of helium and lofted down the hallway like a parachutist, kind of, until she thumped down next to the doorway to the bears’ quarters, just narrowly avoiding a drop right through the open door that would have had her joining the bears, not good, but by spread-eagling she did avoid it, and quickly she closed and locked the enclosure door","name":"Amelia"}]},{"chapter":40,"name":"","length":249,"links":[]},{"chapter":41,"name":"Inspector Gen","length":26922,"links":[{"index":4263,"str":"Vinson’s earlier work for Adirondack Investing, when he had worked with Larry Jackman, had done that kind of trading, and Rosen and Muttchopf had worked for Adirondack too","name":"Mutt"},{"index":4560,"str":"Rosen and Muttchopf’s recent work for Alban Albany had gotten them paid forty thousand dollars apiece","name":"Mutt"},{"index":5152,"str":"On the night Rosen and Muttchopf had disappeared, Sean said, there had been a strange event in the Chicago Mercantile Exchange","name":"Mutt"},{"index":7004,"str":"Or, suppose Vinson was behind Morningside’s offer on the building, and Rosen and Muttchopf had found out about it, or somehow interfered with it","name":"Mutt"},{"index":8539,"str":"The two cousins apparently had lived in the same house for a time, after Jeff’s childhood home got inundated in the Second Pulse","name":"Jeff"},{"index":0,"str":"Inspector Gen walked the skyways to work","name":"Gen"},{"index":1022,"str":"Striking out wildly, three big street cops trying to subdue the person, Gen wasn’t sure about gender here","name":"Gen"},{"index":1406,"str":"Gen winced as the cuffed hands together swung into the ribs of one of the cops","name":"Gen"},{"index":1907,"str":"“To the hospital,” Gen said, but of course they were already headed that way, and only nodded before disappearing down the hall","name":"Gen"},{"index":2070,"str":"“Does anyone know what that was about?” Gen called to those down the hall, minding other business","name":"Gen"},{"index":2681,"str":"Gen thought it over","name":"Gen"},{"index":2895,"str":"When they were seated Gen regarded her whiteboard unenthusiastically","name":"Gen"},{"index":3749,"str":"Olmstead brought her up to date on what he had found in the datasphere, and Gen drew things on the whiteboard, just to help her see: initials, Xs and Os, arrows here and there, lines solid or dotted","name":"Gen"},{"index":5718,"str":"Olmstead had also, at Gen’s behest, been looking into the bid on the Met tower that had so troubled Charlotte","name":"Gen"},{"index":6013,"str":"Gen marked up her whiteboard","name":"Gen"},{"index":6443,"str":"It was a relief to think that neither Charlotte nor Vlade had a good reason to be involved in the disappearance, but Gen was suspicious of her relief","name":"Gen"},{"index":8252,"str":"Only four dimensions! When there are clearly six! Which Gen would refuse to ask him about","name":"Gen"},{"index":11038,"str":"Gen’s cynicism about the amphibious and their speakeasies and bathhouses had hardened in recent years; too many things had changed, too many crimes committed, but she could reach down to that kernel of nostalgia for the old days if she tried hard enough","name":"Gen"},{"index":12009,"str":"The speakeasy door at the bottom of the stairs was in the classic style, and Gen rapped on it using the old code for the submarine gang she had been part of over in Hoboken, thirty years before","name":"Gen"},{"index":12304,"str":"“Ellie is expecting me,” Gen said to the doorman, which was not true except in the sense that it was permanently true","name":"Gen"},{"index":12736,"str":"“Water,” Gen said, to be annoying","name":"Gen"},{"index":12934,"str":"Ellie sat down at her corner table and Gen joined her","name":"Gen"},{"index":13342,"str":"Always she made an attempt to overawe Gen with her etched pale beauty, and Gen had to allow that it took an effort to keep this from working","name":"Gen"},{"index":13588,"str":"Gen knew all that and yet it was still hard not to feel a little frumpy","name":"Gen"},{"index":13930,"str":"And knowing Ellie was here meant Gen knew what was going on underwater","name":"Gen"},{"index":14321,"str":"“I heard there’s bad stuff getting sold in Kips Bay,” Gen said, “and I came over to check it out","name":"Gen"},{"index":14480,"str":"This was too direct, as Gen well knew","name":"Gen"},{"index":14591,"str":"Ellie finished pouting at this brushback fastball, then said, “I know what you mean, Gen, but it isn’t any of us","name":"Gen"},{"index":14793,"str":"The room was in a Faraday box with a magnetic charge that would scramble any recorders, and Gen didn’t have one anyway","name":"Gen"},{"index":15039,"str":"Gen nodded to confirm this, and Ellie leaned forward and said, “There’s a group from uptown putting this shit out, I think to try to wreck the feel down here","name":"Gen"},{"index":15792,"str":"“Real estate?” Gen said","name":"Gen"},{"index":17077,"str":"“This is happening to you,” Gen said","name":"Gen"},{"index":17168,"str":"New York is New York, Gen","name":"Gen"},{"index":17237,"str":"“Mildew,” Gen suggested","name":"Gen"},{"index":17665,"str":"Gen nodded","name":"Gen"},{"index":18017,"str":"Gen could only nod","name":"Gen"},{"index":18204,"str":"Gen shook her head","name":"Gen"},{"index":18648,"str":"Gen gestured at Ezra and Claire, still knocking around the same number of balls on the table","name":"Gen"},{"index":18853,"str":"Shit at Ping-Pong too, Gen presumed","name":"Gen"},{"index":19004,"str":"“I’m going to watch some water sumo,” Gen told them","name":"Gen"},{"index":20205,"str":"All very familiar to Gen","name":"Gen"},{"index":20678,"str":"Gen sat by Ellie and enjoyed the vibe, the friendly hellos, “Oh she’s back,” “Coming back to Mama,” that kind of thing","name":"Gen"},{"index":20799,"str":"“Please, Gen-gen, get back in!”","name":"Gen"},{"index":20917,"str":"“They’ll be here in a second,” Ellie said to Gen","name":"Gen"},{"index":20967,"str":"Gen nodded","name":"Gen"},{"index":22350,"str":"This was the norm at Ellie’s or in any speakeasy bathhouse, so Gen found it all familiar and reassuring","name":"Gen"},{"index":22455,"str":"Ezra and Claire were looking a little round-eyed; they were clearly not denizens of the deep like Gen had once been","name":"Gen"},{"index":22656,"str":"The ref asked if Gen would preside","name":"Gen"},{"index":23346,"str":"Gen took the wand from Cy, the usual ref, who was wearing a red eyepatch tonight, and clicked it to turn on the light","name":"Gen"},{"index":23826,"str":"Gen had been a champion in her time, and she felt a little ghost of the buzz as she watched the two wrestlers settle in","name":"Gen"},{"index":24275,"str":"Gen said, “Go!” and they approached each other, shook hands, moved back","name":"Gen"},{"index":24398,"str":"In some forms of the game you had to keep your head above water, but full immersion had become standard back in Gen’s time, so now these two had sucked air and were down there looking at each other underwater","name":"Gen"},{"index":25014,"str":"Ginger was thrown out of the circle, and Gen called the toss to cheers","name":"Gen"},{"index":25526,"str":"People loved to see women fight, Gen liked it herself","name":"Gen"},{"index":25742,"str":"This was what Gen would have done in her own good youth","name":"Gen"},{"index":26216,"str":"But sumo was about mass staying put, so defense was always king, and queen too, and it didn’t take long for Diane to slip to the side, go deep again, wedge under, and shove off the bottom and catch Ginger right in the midriff and carry her out, Ginger just clearing the circle before Diane did, by about a foot Gen judged, the left foot, and the cameras confirmed it","name":"Gen"},{"index":26600,"str":"Both of them stood and shook hands, first with each other and then with Gen, and Gen was pleased to see they were happy to have her there","name":"Gen"},{"index":6082,"str":"The Met tower was a blue box, with Charlotte Armstrong on one side of it and Vlade Marovich on the other","name":"Vlade"},{"index":6387,"str":"Needed to investigate all Vlade’s employees in the Met","name":"Vlade"},{"index":6443,"str":"It was a relief to think that neither Charlotte nor Vlade had a good reason to be involved in the disappearance, but Gen was suspicious of her relief","name":"Vlade"},{"index":5718,"str":"Olmstead had also, at Gen’s behest, been looking into the bid on the Met tower that had so troubled Charlotte","name":"Charlotte"},{"index":6082,"str":"The Met tower was a blue box, with Charlotte Armstrong on one side of it and Vlade Marovich on the other","name":"Charlotte"},{"index":6443,"str":"It was a relief to think that neither Charlotte nor Vlade had a good reason to be involved in the disappearance, but Gen was suspicious of her relief","name":"Charlotte"}]},{"chapter":42,"name":"","length":255,"links":[]},{"chapter":43,"name":"Mutt and Jeff","length":5826,"links":[{"index":146,"str":"I’m so sorry I got you mixed up with this Mutt","name":"Mutt"},{"index":1462,"str":"Mutt wipes his chin with a napkin","name":"Mutt"},{"index":1497,"str":"“More,” Mutt says","name":"Mutt"},{"index":1593,"str":"Mutt is spooning water into his mouth","name":"Mutt"},{"index":2065,"str":"Mutt shifts back to water","name":"Mutt"},{"index":2839,"str":"“Harassing someone to do the right thing,” Mutt suggests","name":"Mutt"},{"index":3479,"str":"“You’re a pretty great cassandra,” Mutt mentions","name":"Mutt"},{"index":4463,"str":"That’s what they’re doing, Mutt","name":"Mutt"},{"index":5375,"str":"Mutt tries to tuck a last spoonful of maple water in the corner of his mouth, then wipes his chin again and pulls the blanket up over his chest","name":"Mutt"},{"index":0,"str":"Jeff? Are you okay?”","name":"Jeff"},{"index":943,"str":"Jeff shakes his head","name":"Jeff"},{"index":1377,"str":"Jeff sips some water, maybe a tablespoon of it","name":"Jeff"},{"index":1572,"str":"Jeff nods, sips more","name":"Jeff"},{"index":1632,"str":"After this works for a while, he dips the spoon into the little waxed box of maple syrup and feeds Jeff some of that","name":"Jeff"},{"index":1750,"str":"Jeff chokes a little, nods, sits up, and takes in several more spoonfuls of maple syrup","name":"Jeff"},{"index":2092,"str":"After a while Jeff holds a glass of water on his stomach, raises it to sip by himself","name":"Jeff"},{"index":3650,"str":"Now Jeff’s rant sinks to a hoarse whisper","name":"Jeff"},{"index":4969,"str":"“Jeff, stop now","name":"Jeff"},{"index":5349,"str":"Happily Jeff falls asleep","name":"Jeff"}]},{"chapter":44,"name":"","length":148,"links":[]},{"chapter":45,"name":"Stefan and Roberto","length":9561,"links":[{"index":2692,"str":"Vlade knew now that they slept in their boat under his dock, and he often gave them food, also a free charge for their boat’s battery, so they could gurgle over to the wreck rather than row, always taking canals not cordoned off by the water police","name":"Vlade"},{"index":4275,"str":"So the next day they dashed into the Met’s kitchen and snatched a loaf of bread fresh out of the oven, Vlade looking the other way, as he did all the time now","name":"Vlade"},{"index":4489,"str":"Vlade did the same thing for the bacino’s cats","name":"Vlade"},{"index":8535,"str":"He tapped their wristpad and called Vlade","name":"Vlade"},{"index":8578,"str":"Vlade picked up, thank God, and Stefan quickly explained the situation to him","name":"Vlade"},{"index":8657,"str":"“A diving bell?” Vlade repeated, catching the essence of the problem","name":"Vlade"},{"index":9173,"str":"“You kids!” Vlade said sharply","name":"Vlade"},{"index":9300,"str":"Again Vlade was incredulous","name":"Vlade"},{"index":0,"str":"Stefan and Roberto were glad to see the old man settled into the Met tower’s farm","name":"Stefan"},{"index":5037,"str":"Roberto clawed into his wetsuit and Stefan helped him get the diving gear on","name":"Stefan"},{"index":5115,"str":"All was good when Stefan said, “I still don’t see how we’re going to dig down far enough","name":"Stefan"},{"index":5439,"str":"Stefan shook his head","name":"Stefan"},{"index":5615,"str":"Roberto hopped over the side and Stefan lifted the bell over the side and onto him","name":"Stefan"},{"index":5913,"str":"High tide again, so quite a ways down there, which worried Stefan","name":"Stefan"},{"index":6707,"str":"Suddenly Stefan saw that the oxygen tube was taut but the rope was slack","name":"Stefan"},{"index":7167,"str":"“Oh no!” Stefan shouted","name":"Stefan"},{"index":7291,"str":"Stefan tugged on it three times, then shouted down it, though he knew it wasn’t going to convey the sound of his voice to the bottom","name":"Stefan"},{"index":7749,"str":"Again Stefan shouted Roberto’s name, again he tugged three times on the tube, but now gently, as he was scared of pulling it out from under the edge down there","name":"Stefan"},{"index":8148,"str":"The wind was blowing Stefan upstream hard enough that the oxygen tube was stretching flat over the side of the boat","name":"Stefan"},{"index":8323,"str":"Stefan started the motor and hummed back to the buoy, reached over the side and grabbed it","name":"Stefan"},{"index":8578,"str":"Vlade picked up, thank God, and Stefan quickly explained the situation to him","name":"Stefan"},{"index":8734,"str":"“No time for that,” Stefan pleaded, “we’ll tell you later, but can you come and help pull him up? He’s only got about an hour’s air in the air tank, and then I’ll have to change tanks, and I’ve only got one spare","name":"Stefan"},{"index":9283,"str":"Stefan told him","name":"Stefan"},{"index":9362,"str":"“Just come help and we’ll tell you,” Stefan promised","name":"Stefan"},{"index":0,"str":"Stefan and Roberto were glad to see the old man settled into the Met tower’s farm","name":"Roberto"},{"index":3256,"str":"“I don’t think that was his houseboat,” said Roberto","name":"Roberto"},{"index":5037,"str":"Roberto clawed into his wetsuit and Stefan helped him get the diving gear on","name":"Roberto"},{"index":5207,"str":"“We’ll just keep at it,” Roberto said","name":"Roberto"},{"index":5615,"str":"Roberto hopped over the side and Stefan lifted the bell over the side and onto him","name":"Roberto"},{"index":5698,"str":"He could just see Roberto under the clear plastic, rocking the bell to the side to let a little air out from under it","name":"Roberto"},{"index":5817,"str":"A fart of bubbles burst the surface, and then Roberto and the bell were drifting to the bottom","name":"Roberto"},{"index":7102,"str":"Roberto was down there and there was no way to pull him back up","name":"Roberto"},{"index":7425,"str":"For now Roberto had air, but when the gas bottle ran out (and the spare bottle too, there under the thwart) there would still be no way to raise the bell","name":"Roberto"},{"index":7580,"str":"Possibly Roberto could push up one edge of it, duck under the side and swim up to the surface","name":"Roberto"},{"index":7749,"str":"Again Stefan shouted Roberto’s name, again he tugged three times on the tube, but now gently, as he was scared of pulling it out from under the edge down there","name":"Roberto"}]},{"chapter":46,"name":"","length":144,"links":[]},{"chapter":47,"name":"Vlade","length":12699,"links":[{"index":144,"str":"What he needed most was a fast boat, and right as he reached the dock he saw Franklin Garr waiting there for Su to drop his little hydrofoil out of the rafters where Vlade had stashed it","name":"Franklin"},{"index":10593,"str":"Franklin nodded once","name":"Franklin"},{"index":10728,"str":"“What should we do with their diving bell?” Franklin asked Vlade","name":"Franklin"},{"index":0,"str":"Vlade hustled up the stairs to the boathouse dock thinking about what he might need","name":"Vlade"},{"index":144,"str":"What he needed most was a fast boat, and right as he reached the dock he saw Franklin Garr waiting there for Su to drop his little hydrofoil out of the rafters where Vlade had stashed it","name":"Vlade"},{"index":369,"str":"“Hey,” Vlade said, “I need your boat","name":"Vlade"},{"index":779,"str":"Vlade shrugged, wondering how he would do it if he had to grab this guy’s boat from him","name":"Vlade"},{"index":1799,"str":"The canal was crowded this time of day, but he was good at dodging through the crowd, and for once he had an excuse to do it, so he shot the boat over wakes and through gaps between barges and kayaks and vapos and rowboats and gondolas, gaps smaller than Vlade would have dared to attempt","name":"Vlade"},{"index":2366,"str":"Vlade marveled at the speed with which the UN building shot past on their left","name":"Vlade"},{"index":2695,"str":"Despite himself Vlade was impressed, and almost feeling a tiny glow of relief through the knot in his stomach","name":"Vlade"},{"index":3563,"str":"Ordinarily Vlade would have thought it much too fast, but given the circumstances he was happy the man was reckless","name":"Vlade"},{"index":3727,"str":"Vlade held his breath as they crossed over some dark spots in the blue, but they passed safely","name":"Vlade"},{"index":4253,"str":"“How far down?” Vlade asked","name":"Vlade"},{"index":4357,"str":"Vlade sighed","name":"Vlade"},{"index":6002,"str":"At first it obviously resisted them, so much that Vlade was amazed the boys had been able to pull each other up alone","name":"Vlade"},{"index":6227,"str":"Then the two in the boat got it going, and Vlade put his face mask back in position and dove again, to help Roberto out from under the bell’s edge and into the boat","name":"Vlade"},{"index":6842,"str":"Vlade nodded at him, pointed up, and hauled him down into the water, under the bell’s edge, and up to the surface","name":"Vlade"},{"index":7136,"str":"Vlade crawled up over the side of the zodiac, never an easy move, but soon enough he flopped over the fat rubber tube into the cockpit","name":"Vlade"},{"index":7420,"str":"Vlade pulled off his own face mask and unclipped from his tank and got out of his gear","name":"Vlade"},{"index":7672,"str":"“Jump up and draw us a bowl of the hottest water you have,” Vlade said","name":"Vlade"},{"index":8341,"str":"“Have you got a towel?” Vlade asked Stefan","name":"Vlade"},{"index":8441,"str":"Vlade took it from him and began to dry Roberto’s head off, at the same time roughing him up a little to get his circulation going faster","name":"Vlade"},{"index":8699,"str":"Vlade tried to clear his mind enough to recall standard practice in the city","name":"Vlade"},{"index":9023,"str":"“Did the oxygen keep flowing to you the whole time?” Vlade asked Roberto","name":"Vlade"},{"index":9439,"str":"Garr stepped over the gunwales into the zodiac’s cockpit with minimal spillage from the bowl in his hands, and Vlade took the bowl and scooped water out with his hand, scalding his fingers more with the contrast of temperatures than the water’s actual heat, and dripped some of it onto Roberto’s chest","name":"Vlade"},{"index":9795,"str":"Vlade was past the moment of his flashback now, back in the present moment with this kid, who was going to be all right","name":"Vlade"},{"index":9917,"str":"“Slowly,” Vlade said, and had Stefan continue to dry Roberto’s hair with the towel","name":"Vlade"},{"index":10508,"str":"“Okay,” Vlade said after a while","name":"Vlade"},{"index":10728,"str":"“What should we do with their diving bell?” Franklin asked Vlade","name":"Vlade"},{"index":10831,"str":"As Garr was in his cockpit piloting them, Vlade sat back and got himself between Roberto and the wind","name":"Vlade"},{"index":11050,"str":"Vlade shook his head","name":"Vlade"},{"index":11226,"str":"“Ah come on,” Vlade said","name":"Vlade"},{"index":12441,"str":"“I’m sure,” Vlade said","name":"Vlade"},{"index":12683,"str":"“Thanks, Vlade","name":"Vlade"},{"index":420,"str":"“Sorry, but those kids Roberto and Stefan are in trouble up in the south Bronx","name":"Stefan"},{"index":1566,"str":"“That’s what Stefan said","name":"Stefan"},{"index":4101,"str":"They pulled up next to Stefan, still standing in the zodiac, looking relieved","name":"Stefan"},{"index":5822,"str":"“Did you see him?” Stefan asked anxiously","name":"Stefan"},{"index":5964,"str":"Stefan and Garr hauled up on the rope","name":"Stefan"},{"index":8341,"str":"“Have you got a towel?” Vlade asked Stefan","name":"Stefan"},{"index":8385,"str":"Stefan nodded and got it from a locker under the thwart","name":"Stefan"},{"index":9917,"str":"“Slowly,” Vlade said, and had Stefan continue to dry Roberto’s hair with the towel","name":"Stefan"},{"index":10310,"str":"Stefan finished drying his head off","name":"Stefan"},{"index":10614,"str":"“I can’t believe I’m towing you guys home again,” he said to Stefan and Roberto","name":"Stefan"},{"index":11541,"str":"Stefan said, “We did because Mr","name":"Stefan"},{"index":12346,"str":"“That’s what I said,” Stefan said after a silence","name":"Stefan"},{"index":420,"str":"“Sorry, but those kids Roberto and Stefan are in trouble up in the south Bronx","name":"Roberto"},{"index":5622,"str":"Presumably Roberto, so he knocked three times on the side of it, then tied the rope on, three loops, after which he tugged hard","name":"Roberto"},{"index":6227,"str":"Then the two in the boat got it going, and Vlade put his face mask back in position and dove again, to help Roberto out from under the bell’s edge and into the boat","name":"Roberto"},{"index":7271,"str":"Roberto lay next to him on the bottom, wet, muddy, his face a brown tinged with blue","name":"Roberto"},{"index":7508,"str":"Then he sat beside Roberto and held his blue little hand","name":"Roberto"},{"index":7773,"str":"” He put his face down to Roberto’s and said, “Roberto, what the hell? You could have died down there!” And suddenly his throat closed up again and he couldn’t say more","name":"Roberto"},{"index":8207,"str":"Roberto was shivering too hard to answer, but he nodded","name":"Roberto"},{"index":8441,"str":"Vlade took it from him and began to dry Roberto’s head off, at the same time roughing him up a little to get his circulation going faster","name":"Roberto"},{"index":9023,"str":"“Did the oxygen keep flowing to you the whole time?” Vlade asked Roberto","name":"Roberto"},{"index":9097,"str":"Roberto shook his head, then with difficulty said, “The bell edge squished it","name":"Roberto"},{"index":9439,"str":"Garr stepped over the gunwales into the zodiac’s cockpit with minimal spillage from the bowl in his hands, and Vlade took the bowl and scooped water out with his hand, scalding his fingers more with the contrast of temperatures than the water’s actual heat, and dripped some of it onto Roberto’s chest","name":"Roberto"},{"index":9917,"str":"“Slowly,” Vlade said, and had Stefan continue to dry Roberto’s hair with the towel","name":"Roberto"},{"index":10084,"str":"Roberto kept shivering, with occasional spasms of extra shuddering, but shivering was good; there was a point where you got too cold to shiver, very hard to come back from","name":"Roberto"},{"index":10614,"str":"“I can’t believe I’m towing you guys home again,” he said to Stefan and Roberto","name":"Roberto"},{"index":10831,"str":"As Garr was in his cockpit piloting them, Vlade sat back and got himself between Roberto and the wind","name":"Roberto"},{"index":10988,"str":"Roberto gulped","name":"Roberto"},{"index":11170,"str":"“It’s the Hussar,” Roberto said","name":"Roberto"},{"index":12397,"str":"“I had a plan,” Roberto muttered miserably","name":"Roberto"},{"index":12464,"str":"He tousled Roberto’s turban","name":"Roberto"}]},{"chapter":48,"name":"","length":790,"links":[]},{"chapter":49,"name":"Citizen","length":13125,"links":[]},{"chapter":50,"name":"","length":122,"links":[]},{"chapter":51,"name":"Franklin","length":16935,"links":[{"index":205,"str":"“For a very important date,” Vlade added heavily","name":"Vlade"},{"index":4396,"str":"“Vlade requisitioned me as I was leaving the Met, we had to zip up to the Bronx and rescue those two little squeakers who saved the old man, it was their turn to be saved","name":"Vlade"},{"index":4568,"str":"” And I explained how they had managed to get Roberto stuck on the bottom of the south Bronx, with Stefan up in their boat holding his oxygen bottle and nothing else","name":"Stefan"},{"index":270,"str":"Garr,” Roberto said","name":"Roberto"},{"index":4568,"str":"” And I explained how they had managed to get Roberto stuck on the bottom of the south Bronx, with Stefan up in their boat holding his oxygen bottle and nothing else","name":"Roberto"}]},{"chapter":52,"name":"","length":157,"links":[]},{"chapter":53,"name":"Charlotte","length":30747,"links":[{"index":108,"str":"Charlotte didn’t want to discuss it in a general meeting of all the co-op members, which she knew was wrong of her, but she didn’t","name":"Charlotte"},{"index":620,"str":"“Oh be quiet,” Charlotte said","name":"Charlotte"},{"index":2435,"str":"Charlotte enjoyed visiting it when it jellyfished by New York, eddying outside the Verrazano Narrows in the big counterclockwise current that curled off the Gulf Stream","name":"Charlotte"},{"index":3744,"str":"This always amazed Charlotte","name":"Charlotte"},{"index":4002,"str":"It was a look Charlotte knew well, the look of her clients, here again staring at her","name":"Charlotte"},{"index":4820,"str":"Charlotte sat down at a table with them and listened to their stories one by one, creating primary refugee documentation","name":"Charlotte"},{"index":6516,"str":"Flying back in over the Brooklyn shallows to the Turtle Bay aircraft carrier anchored next to the UN building, Charlotte sat at the left-side window and marveled at the city in the late-afternoon light","name":"Charlotte"},{"index":7635,"str":"Charlotte shrugged, almost saying I gave at the office, but biting back the words; she didn’t get why they were there","name":"Charlotte"},{"index":8077,"str":"“No way!” Charlotte said, shocked","name":"Charlotte"},{"index":9508,"str":"Charlotte indicated this as quickly and politely as she could, then jumped on the vaporetto that mercifully gurgled into the dock headed down Park, just as Charlotte’s interlocutors were waxing eloquent with desperate pleadings","name":"Charlotte"},{"index":9879,"str":"“I will!” Charlotte lied cheerfully","name":"Charlotte"},{"index":11054,"str":"” Charlotte hated the phrase doing due diligence, but this was not the time to mention that","name":"Charlotte"},{"index":11295,"str":"Charlotte said, “I know","name":"Charlotte"},{"index":11912,"str":"Charlotte thought it over","name":"Charlotte"},{"index":12217,"str":"“I’d like to know before we decide,” Charlotte said","name":"Charlotte"},{"index":13317,"str":"“Second,” Charlotte said","name":"Charlotte"},{"index":13380,"str":"So, the next morning Charlotte gritted her teeth and called her ex, Larry Jackman","name":"Charlotte"},{"index":13463,"str":"“Hey Charlotte,” he said","name":"Charlotte"},{"index":14101,"str":"This was one of their hangouts from the old days, so it was with a little lurch that Charlotte agreed","name":"Charlotte"},{"index":15946,"str":"Or so it had seemed to Charlotte","name":"Charlotte"},{"index":16480,"str":"After some years that had mellowed into a kind of rueful nostalgia, and later still, getting together over coffee satisfied a little itch of curiosity in Charlotte, an urge to see how Larry’s story had continued","name":"Charlotte"},{"index":21002,"str":"” Charlotte thought that over","name":"Charlotte"},{"index":21303,"str":"“I don’t know,” Charlotte confessed","name":"Charlotte"},{"index":23331,"str":"Charlotte thought about that","name":"Charlotte"},{"index":23603,"str":"Charlotte laughed too","name":"Charlotte"},{"index":23991,"str":"It would never be quite natural for Charlotte to ask Larry about any of his acquaintances in finance, as she had never taken any interest in them, nor had Larry been inclined to share details of his interactions with them","name":"Charlotte"},{"index":27151,"str":"Charlotte nodded, trying to look like she believed it","name":"Charlotte"},{"index":28413,"str":"Larry being Larry, he managed to make that sound like he saw more than Charlotte would like","name":"Charlotte"},{"index":29990,"str":"Charlotte said something to this effect, and Larry shook his head and chuckled at her","name":"Charlotte"},{"index":30373,"str":"Annoyed, Charlotte said nothing","name":"Charlotte"},{"index":30671,"str":"“We should do this more often,” Charlotte lied","name":"Charlotte"}]},{"chapter":54,"name":"","length":519,"links":[]},{"chapter":55,"name":"Vlade","length":24681,"links":[{"index":0,"str":"Vlade now made a kind of cop’s round of the building every evening after dinner, checking all the security systems and visiting all the rooms lower than the high tide line","name":"Vlade"},{"index":715,"str":"A couple of nights after they had pulled Roberto out of the south Bronx, at the end of his tour of the building, Vlade got off the elevator at the farm and went out to the southeast corner to see how the old man was doing","name":"Vlade"},{"index":1133,"str":"Vlade sat","name":"Vlade"},{"index":1721,"str":"“Beautiful,” Vlade commented","name":"Vlade"},{"index":2005,"str":"“Wow,” Vlade said","name":"Vlade"},{"index":2347,"str":"Vlade laughed","name":"Vlade"},{"index":2709,"str":"Vlade looked for their neighborhood","name":"Vlade"},{"index":3062,"str":"“I’m not sure,” Vlade said","name":"Vlade"},{"index":3179,"str":"Vlade nodded","name":"Vlade"},{"index":3489,"str":"“There’s not one map,” he told Vlade","name":"Vlade"},{"index":4471,"str":"“Holy God,” Vlade said","name":"Vlade"},{"index":4902,"str":"“Okay,” Vlade said","name":"Vlade"},{"index":5324,"str":"“Very nice,” Vlade said","name":"Vlade"},{"index":7553,"str":"“So how come the Brits didn’t recover the gold?” Vlade asked","name":"Vlade"},{"index":7805,"str":"Vlade nodded","name":"Vlade"},{"index":8204,"str":"“Boys?” Vlade said","name":"Vlade"},{"index":8563,"str":"“Where’d you get the diving bell?” Vlade asked","name":"Vlade"},{"index":8994,"str":"Vlade and Hexter gave each other a look","name":"Vlade"},{"index":9034,"str":"“You got to watch out for these guys,” Vlade said","name":"Vlade"},{"index":9666,"str":"This was beginning to sound a little altruistical to Vlade","name":"Vlade"},{"index":10478,"str":"“Really?” Vlade asked","name":"Vlade"},{"index":10527,"str":"Vlade looked at Hexter, who shrugged","name":"Vlade"},{"index":10721,"str":"Vlade sat looking at the map from 1821","name":"Vlade"},{"index":12203,"str":"Vlade decided to take the boys out to Coney Island on his own boat, even though the building’s boat was a bit faster, because he didn’t want this trip on the books","name":"Vlade"},{"index":13498,"str":"Vlade knew that world was out there, but he never went inland; he had never been more than five miles from the ocean in his life","name":"Vlade"},{"index":13773,"str":"The blue Atlantic! Swells rocked the boat, and Vlade had to slow down as he turned left to hug the shore, now marked by a white line of crashing breakers","name":"Vlade"},{"index":13928,"str":"For a half hour they ran southeast just offshore, until they passed Bath Beach, where Vlade headed the boat straight south to Sea Gate, the western end of Coney Island","name":"Vlade"},{"index":14311,"str":"Vlade wondered if the boys might be susceptible to seasickness, but they stood in the cockpit staring around, oblivious to the rocking, which Vlade himself found rather queasy-making","name":"Vlade"},{"index":15039,"str":"From here the coastline looked endless, though Vlade knew for a fact that Coney Island was only about four miles long","name":"Vlade"},{"index":15445,"str":"Ultimately, Vlade thought, you had to accept that the illusion was basically true: the world was huge","name":"Vlade"},{"index":15638,"str":"Vlade laughed to see them","name":"Vlade"},{"index":15767,"str":"“And I thought I was local,” Vlade said","name":"Vlade"},{"index":16043,"str":"Vlade approached the barge","name":"Vlade"},{"index":16070,"str":"It was tall and long, accompanied by a tug that looked small in comparison, though the tug dwarfed Vlade’s boat as they drew alongside","name":"Vlade"},{"index":16206,"str":"There was a dock tied to the barge that Vlade could draw up to, and a crew of dockmen to grab their painter and tie them fast to dock cleats","name":"Vlade"},{"index":16349,"str":"Vlade had called ahead, feeling more nervous than he had felt for many years, and sure enough, there was Idelba now, standing at the back of the group","name":"Vlade"},{"index":16603,"str":"Vlade’s ex-wife, and the one person from his past he still thought about, the only one still alive anyway","name":"Vlade"},{"index":18030,"str":"“Where are you dumping it?” Vlade asked","name":"Vlade"},{"index":18774,"str":"Vlade squinted at the sun","name":"Vlade"},{"index":19129,"str":"“I think we’ll have to do that some other time,” Vlade said, “or else we won’t get back to Manhattan by dinner","name":"Vlade"},{"index":19360,"str":"She had still not met Vlade’s eye, as far as he could tell","name":"Vlade"},{"index":19523,"str":"Now she glanced at Vlade","name":"Vlade"},{"index":19561,"str":"And Vlade knows how well I keep my promises","name":"Vlade"},{"index":19608,"str":"Vlade laughed painfully at that, but when the boys looked alarmed, he said, “No, I’m just laughing because Idelba surprised me there","name":"Vlade"},{"index":20103,"str":"“They almost drowned,” Vlade added despite himself","name":"Vlade"},{"index":20552,"str":"That’s probably why I hung out with Vlade here","name":"Vlade"},{"index":20780,"str":"He glared at Vlade: “You told her!”","name":"Vlade"},{"index":20817,"str":"Vlade shook his head, and Idelba laughed her short harsh laugh","name":"Vlade"},{"index":21468,"str":"“I didn’t know you were into this stuff,” Vlade observed","name":"Vlade"},{"index":21767,"str":"Vlade could see they were developing a crush on Idelba","name":"Vlade"},{"index":21916,"str":"“I was stuck under the bell, but Stefan got Vlade to come out and save me","name":"Vlade"},{"index":21993,"str":"“Good for Vlade","name":"Vlade"},{"index":22009,"str":"” A shadow passed over her face and for a second she wasn’t there with them, and Vlade knew where she was","name":"Vlade"},{"index":22264,"str":"” She regarded them, glanced again at Vlade","name":"Vlade"},{"index":23566,"str":"They were not going to remember not to be disappointed, Vlade could see","name":"Vlade"},{"index":24225,"str":"Everybody was going to be okay compared to Vlade and Idelba","name":"Vlade"},{"index":24385,"str":"Difficult for Vlade to know what Idelba was thinking; she was hard, and he was stunned just to see her again; he had no idea what he was feeling","name":"Vlade"},{"index":937,"str":"No surprise to look in through the hotello’s flap door and find Stefan and Roberto there with him, seated on the floor around a pile of old maps","name":"Stefan"},{"index":5405,"str":"“I like the way the water has waves in it,” Stefan said","name":"Stefan"},{"index":8435,"str":"Stefan said, “We took it to the bottom where Mr","name":"Stefan"},{"index":8639,"str":"“It’s the top of a barge’s grain hopper,” Stefan explained","name":"Stefan"},{"index":10231,"str":"“That’s what I said,” Stefan said","name":"Stefan"},{"index":18602,"str":"“What if sea level rises again?” Stefan asked","name":"Stefan"},{"index":19853,"str":"“Stefan and I are doing a little underwater archaeology in the Bronx, and we think we’ve found a, a find, that we want to dig up, but we’ve been working with just a diving bell, and we can’t do the excavation under it","name":"Stefan"},{"index":21916,"str":"“I was stuck under the bell, but Stefan got Vlade to come out and save me","name":"Stefan"},{"index":22761,"str":"“Wow,” Stefan said","name":"Stefan"},{"index":23496,"str":"“Wow,” Stefan said again","name":"Stefan"},{"index":715,"str":"A couple of nights after they had pulled Roberto out of the south Bronx, at the end of his tour of the building, Vlade got off the elevator at the farm and went out to the southeast corner to see how the old man was doing","name":"Roberto"},{"index":937,"str":"No surprise to look in through the hotello’s flap door and find Stefan and Roberto there with him, seated on the floor around a pile of old maps","name":"Roberto"},{"index":3414,"str":"“We had to tell him,” Roberto said","name":"Roberto"},{"index":4783,"str":"“I want to make a map like this for what’s here now,” Roberto declared","name":"Roberto"},{"index":8224,"str":"“Well,” Roberto said, “it was just a case of one thing leading to another, really","name":"Roberto"},{"index":8533,"str":"“It was great!” Roberto said","name":"Roberto"},{"index":8611,"str":"“We made it,” Roberto said","name":"Roberto"},{"index":10266,"str":"“I think we could have,” Roberto insisted","name":"Roberto"},{"index":12093,"str":"The boys and the old man looked at each other for a while, and then Roberto said, “Okay, sure","name":"Roberto"},{"index":15950,"str":"“Is she about halfway done with it?” Roberto asked","name":"Roberto"},{"index":17725,"str":"“It seems like it could get a lot fuller,” Roberto said","name":"Roberto"},{"index":18849,"str":"“Can we go up with you and see the new beach?” Roberto asked","name":"Roberto"},{"index":19444,"str":"Roberto said, “You have to promise to keep this a secret","name":"Roberto"},{"index":19827,"str":"“Okay then,” Roberto said","name":"Roberto"},{"index":20754,"str":"“What!” Roberto protested","name":"Roberto"},{"index":21875,"str":"“Just last week!” Roberto was explaining","name":"Roberto"},{"index":22781,"str":"“We get it,” Roberto said","name":"Roberto"},{"index":23521,"str":"He and Roberto were both completely smitten","name":"Roberto"}]},{"chapter":56,"name":"","length":501,"links":[]},{"chapter":57,"name":"Amelia","length":22829,"links":[{"index":0,"str":"Amelia, having retaken control of the Assisted Migration, spent the next day or two eating and calming down, with just one camera on, and very little commentary, most of it more suited to a cooking show than animal affairs","name":"Amelia"},{"index":1965,"str":"All this sounded plausible and right to Amelia, so she kept telling Frans to head that way","name":"Amelia"},{"index":2417,"str":"A third faction of ecologists apparently thought they should deposit these bears on South Georgia, so Amelia kept the airship on course to pass within sight of that island as she headed south, just in case","name":"Amelia"},{"index":2989,"str":"As the airship passed to the east of South Georgia, which took most of a day, Amelia found herself glad that she had not been directed to drop the polar bears there; the island was huge, steep, and green where it was not covered by snow and ice, or mantled in cloud that whipped over it in a blown cap that reminded her of the jet stream flying madly over the Himalayas","name":"Amelia"},{"index":4759,"str":"Amelia brought the airship down for a closer look and better images for her show, and from that level they could see streaks of blood on the ice, placental blood for the most part; many of the female Weddell seals, looking like slugs laid out on a sheet of white paper, had recently given birth, and smaller offspring (but not that much smaller) were attached to them, nursing away","name":"Amelia"},{"index":5194,"str":"“Wow, check it out,” Amelia said to her audience","name":"Amelia"},{"index":5880,"str":"As the airship approached the coast, Amelia wondered aloud if they were going to be able to tell where sea ice ended and snow on land began; everything ahead looked white, except for some black cliffs farther inland","name":"Amelia"},{"index":6447,"str":"As they floated over these, Amelia looked down and squealed: there underneath them was a pod of orcas, just a bit blacker than the water itself, with white flashings on their sides, only visible when they arched slightly up and out of the water","name":"Amelia"},{"index":6742,"str":"“Dang!” Amelia said","name":"Amelia"},{"index":7500,"str":"“Oh my God that’s just soo trippy,” Amelia exclaimed","name":"Amelia"},{"index":7553,"str":"This was one of her signature exclamations, controversial back in the studio, as being either a cloying old-fashioned cliché or else an endearing Amelia-ism, but in any case Amelia couldn’t help it, it was just what she felt","name":"Amelia"},{"index":8906,"str":"This was their destination, Amelia was told","name":"Amelia"},{"index":9181,"str":"“The black rock is basalt, the red rock is dolerite,” Amelia repeated from her studio feed","name":"Amelia"},{"index":11193,"str":"Having looked at the potential winter dens from as close as the airship could come, Amelia turned back toward the coast","name":"Amelia"},{"index":11504,"str":"“Just hold your horses!” Amelia called down the hall","name":"Amelia"},{"index":12808,"str":"“Amelia, you can’t be out there when the bears are released","name":"Amelia"},{"index":13839,"str":"“You are the one controlling the door, Amelia","name":"Amelia"},{"index":14014,"str":"Between the wind pouring into the door and the bears pouring out, the ship got quite a shaking, and Amelia squealed","name":"Amelia"},{"index":14604,"str":"And yet they didn’t look alarmed by the bears, as why should they? For one thing the bears were now nearly invisible, such that Amelia only caught glimpses of them, like a crab made of black claws, or a pair of black eyes like the coal eyes of a snowman, glancing back her way and then winking out","name":"Amelia"},{"index":16199,"str":"Of course each of them was tagged with a radio transmitter and a few minicams, so Amelia’s viewers would get to see them live their lives on her show","name":"Amelia"},{"index":16447,"str":"Amelia’s Animals was a very popular spin-off site in the cloud","name":"Amelia"},{"index":16656,"str":"“What’s wrong?” Amelia asked","name":"Amelia"},{"index":16904,"str":"She tapped away, and then Amelia was looking at the Antarctic peninsula and the sea ice: then there was a bright white light, and nothing more","name":"Amelia"},{"index":17543,"str":"“What?” Amelia cried","name":"Amelia"},{"index":17855,"str":"“Fuck them!” Amelia shouted, and started to beat the table leg beside her, then to cry","name":"Amelia"},{"index":17967,"str":"No response from Nicole, and Amelia suddenly realized they were still transmitting her response to her audience","name":"Amelia"},{"index":18182,"str":"The next day Amelia stood in front of one of her cameras and turned it on","name":"Amelia"}]},{"chapter":58,"name":"","length":126,"links":[]},{"chapter":59,"name":"Citizen","length":5542,"links":[]},{"chapter":60,"name":"","length":190,"links":[]},{"chapter":61,"name":"Inspector Gen","length":19786,"links":[{"index":0,"str":"Gen sometimes wondered if the patterns she thought she saw caused her to send her people out and make the patterns come into existence","name":"Gen"},{"index":334,"str":"Sometimes Claire would come back from her night classes talking about the dialectic, and what she said sounded a bit like Gen’s thinking","name":"Gen"},{"index":1022,"str":"So Gen was baffled as she reflected on these matters while walking the skybridges of the drowned city from station to station, from problem to problem","name":"Gen"},{"index":1753,"str":"Mayor Estaban was hosting some kind of ceremony for visiting mayors from inland cities, apparently, and Inspector Gen had decided to attend and wave the NYPD flag","name":"Gen"},{"index":1918,"str":"This crowd was not Gen’s crowd","name":"Gen"},{"index":4437,"str":"So now Gen clomped up the broad staircase to the mezzanine, moving slowly, like a beat cop with sore dogs","name":"Gen"},{"index":4860,"str":"When the mayor and her retinue swept in, Gen stayed put and watched the throng gather around them and then dissipate, allowing the mayor to make her rounds","name":"Gen"},{"index":5323,"str":"She reminded Gen of Amelia Black in some ways—in that easy assumption of fame","name":"Gen"},{"index":5729,"str":"When she saw Gen she came immediately to her, as if on a tractor beam to her favorite person in the room, or even the most important person in the room","name":"Gen"},{"index":5881,"str":"Gen almost smiled as she acknowledged that yes, Galina Estaban was the best ever at making people feel good","name":"Gen"},{"index":6118,"str":"But in this case Gen knew it was all an act","name":"Gen"},{"index":6163,"str":"Gen had nailed one of Estaban’s favorite aides taking kickbacks from an uptown developer, and really it was obvious from the proximities involved that Estaban had to have known about it","name":"Gen"},{"index":6350,"str":"Galina hadn’t liked having to accept her aide’s hasty resignation and had retaliated with some hammering of Gen’s support at police headquarters, then some disabling blows at their cloud infrastructure, which was ugly revenge indeed; NYPD had been materially harmed","name":"Gen"},{"index":7864,"str":"Gen was used to stonewalling, from them and from everyone, and in other situations she might have undermined their stone wall a little, but this was not a time for that","name":"Gen"},{"index":8182,"str":"Gen shrugged back at John, trying to indicate with her stare that she didn’t expect any minion help at any time","name":"Gen"},{"index":8721,"str":"Gen stared at the mayor","name":"Gen"},{"index":9363,"str":"Gen did that now, while hoping that Claire had some of her implants in the mayor’s office here watching the minions on hand","name":"Gen"},{"index":10224,"str":"Estaban blinked, processing both the question and the fact that Gen had asked it at a reception like this one","name":"Gen"},{"index":10859,"str":"She surveyed the minions herself, trying to gauge the room; they were very attuned to the mayor’s mood, and now they were a little freaked, and not looking Gen’s way","name":"Gen"},{"index":11026,"str":"With a sudden onset of lethemlucidity Gen saw the power structure of the city with x-ray vision, all atremble with force fields like magnetic lines emanating from the gorgeous mayor","name":"Gen"},{"index":11209,"str":"Gen had broken the glass over some kind of psychic alarm bell, and now it was ringing","name":"Gen"},{"index":11654,"str":"Gen had the impression he spent far more time here than in his room, which was basically just a place to sleep","name":"Gen"},{"index":12615,"str":"That would represent disaster for him, it seemed to Gen","name":"Gen"},{"index":13405,"str":"Gen put her pad on his desk and he transferred some files into it","name":"Gen"},{"index":14061,"str":"“One of your favorite residents,” Gen guessed","name":"Gen"},{"index":14779,"str":"Gen was a well-known figure and lots of friends and acquaintances came up to greet her; it had been a while since she had made one of the monthlies","name":"Gen"},{"index":15908,"str":"Gentrification my ass","name":"Gen"},{"index":16153,"str":"I know, Gen kept saying","name":"Gen"},{"index":19019,"str":"But they dragged her along, and a punk power band with a gaggle of vocalists was firing through Lou Reed’s “Heroin” like it was the national anthem, which down here maybe it was, and Gen wanted to object that the drug being celebrated was more likely to create a flow state than this supercharged nails-on-blackboard style they were indulging in, but what did she know really","name":"Gen"},{"index":13526,"str":"Franklin Garr, a resident","name":"Franklin"},{"index":18556,"str":"A Ben Franklin pun, young Franklin Garr had recently informed her with a look of amused pride","name":"Franklin"},{"index":11442,"str":"When she got back to her apartment in the Met she changed clothes, went down to Vlade’s basement room, buzzed the door","name":"Vlade"},{"index":12516,"str":"More typically new ownership would hire new management, in which case Vlade would be out of a job","name":"Vlade"},{"index":13709,"str":"Eyebrows bunched as he drove home his request to Vlade","name":"Vlade"},{"index":13829,"str":"“On its way,” Vlade said heavily, toggling his boathouse dashboard","name":"Vlade"},{"index":14108,"str":"Vlade smiled","name":"Vlade"},{"index":18137,"str":"She could see why Vlade and Charlotte didn’t make many of these meetings","name":"Vlade"},{"index":5323,"str":"She reminded Gen of Amelia Black in some ways—in that easy assumption of fame","name":"Amelia"},{"index":5402,"str":"She was like Amelia’s Latina older sister, the one who had gotten good grades and even enjoyed studying","name":"Amelia"},{"index":18137,"str":"She could see why Vlade and Charlotte didn’t make many of these meetings","name":"Charlotte"}]},{"chapter":62,"name":"","length":654,"links":[]},{"chapter":63,"name":"Franklin","length":29958,"links":[{"index":19715,"str":"“Franklin Garr","name":"Franklin"}]},{"chapter":64,"name":"","length":825,"links":[]},{"chapter":65,"name":"Mutt and Jeff","length":11640,"links":[{"index":1315,"str":"Mutt doesn’t know what to say to this","name":"Mutt"},{"index":4775,"str":"Mutt nods, looking pleased","name":"Mutt"},{"index":4949,"str":"“But how?” Mutt asks, nudging the tide","name":"Mutt"},{"index":5610,"str":"Mutt sighs","name":"Mutt"},{"index":5765,"str":"Mutt doesn’t know what to say, but on the other hand he has to say something, to keep Jeff going","name":"Mutt"},{"index":8066,"str":"No, Muttnik","name":"Mutt"},{"index":8646,"str":"After a while Mutt says, “Jeff? Are you awake?”","name":"Mutt"},{"index":11546,"str":"And then, because Jeff is asleep and cannot see it, Mutt puts his face in his hands and cries","name":"Mutt"},{"index":0,"str":"Jeff, are you awake?”","name":"Jeff"},{"index":1353,"str":"He sits by his friend’s bed, reaches over and holds Jeff’s hand","name":"Jeff"},{"index":1479,"str":"” Jeff eyes his friend","name":"Jeff"},{"index":5622,"str":"“I know!” Jeff says","name":"Jeff"},{"index":5765,"str":"Mutt doesn’t know what to say, but on the other hand he has to say something, to keep Jeff going","name":"Jeff"},{"index":5862,"str":"So he says, “Jeff, these are laws you’re talking about here","name":"Jeff"},{"index":7263,"str":"Jeff is silent for a long time","name":"Jeff"},{"index":8415,"str":"Again Jeff is silent","name":"Jeff"},{"index":8646,"str":"After a while Mutt says, “Jeff? Are you awake?”","name":"Jeff"},{"index":8694,"str":"After a while Jeff rouses","name":"Jeff"},{"index":8810,"str":"” Long pause; possibly Jeff is weeping here","name":"Jeff"},{"index":10227,"str":"“Why didn’t anyone live there before?” Jeff asks from out of his sleep","name":"Jeff"},{"index":11354,"str":"Hey, Jeff? Jeff? Well, the end, I guess","name":"Jeff"},{"index":11397,"str":"Because Jeff is now lying there peacefully snoring","name":"Jeff"},{"index":11546,"str":"And then, because Jeff is asleep and cannot see it, Mutt puts his face in his hands and cries","name":"Jeff"}]},{"chapter":66,"name":"","length":340,"links":[]},{"chapter":67,"name":"Stefan and Roberto","length":20822,"links":[{"index":8532,"str":"Idelba said, “Gentlemen, the dredge is gonna start sucking up whatever’s down there","name":"Gen"},{"index":0,"str":"Stefan and Roberto were subdued and even apprehensive on the day they joined Vlade and his friend Idelba on her tugboat","name":"Vlade"},{"index":1446,"str":"Vlade came up with Idelba after they had cast off, and the tug’s pilot, a skinny black man named Thabo, pushed the throttle forward and they shoved upriver","name":"Vlade"},{"index":1761,"str":"“No chance of hiding this baby,” Vlade remarked when he saw the looks on the boys’ faces","name":"Vlade"},{"index":2326,"str":"“Both of them,” Vlade added","name":"Vlade"},{"index":4844,"str":"Idelba and Vlade and Mr","name":"Vlade"},{"index":4868,"str":"Hexter came down to the deck, and Vlade helped Idelba and Thabo deploy the dredge tube over the side","name":"Vlade"},{"index":4970,"str":"Vlade got Roberto and Stefan involved in moving the segments of the tube to the rear and latching them onto the long snake they were making","name":"Vlade"},{"index":5740,"str":"“Here, come check this out,” Vlade said to the boys","name":"Vlade"},{"index":6286,"str":"“I guess we left it behind after Vlade got me out of it","name":"Vlade"},{"index":6933,"str":"Vlade was helping them on the sonar and radar, obviously very comfortable with all the gear","name":"Vlade"},{"index":9585,"str":"The boys stood with Vlade and Mr","name":"Vlade"},{"index":10160,"str":"Vlade put on rubber gloves that went up to his elbow, then a dust mask over his nose, and began to finger through the mud in the box","name":"Vlade"},{"index":10546,"str":"Thabo and Vlade uncoupled the last section of tubing and began rooting around in the last tube","name":"Vlade"},{"index":13221,"str":"Thabo and Vlade delinked the tube from the box, and they began to dig in the muck caught on the tube’s filter","name":"Vlade"},{"index":13501,"str":"Vlade tried to lift it out by himself and couldn’t","name":"Vlade"},{"index":15465,"str":"Vlade just smiled at them, shaking his head","name":"Vlade"},{"index":17372,"str":"“So whose are they?” Roberto asked, looking at Vlade","name":"Vlade"},{"index":17426,"str":"Vlade saw the nature of his look and laughed","name":"Vlade"},{"index":18224,"str":"“I could ask Charlotte,” Vlade said","name":"Vlade"},{"index":18423,"str":"Before they reached Twenty-sixth, Thabo said something to Idelba, who called Vlade over to the scanning screens","name":"Vlade"},{"index":18874,"str":"“Subway entry?” Vlade asked","name":"Vlade"},{"index":20052,"str":"Idelba and Vlade regarded each other","name":"Vlade"},{"index":20090,"str":"“Let’s get it into the Met,” Vlade suggested","name":"Vlade"},{"index":20523,"str":"Vlade nodded","name":"Vlade"},{"index":18224,"str":"“I could ask Charlotte,” Vlade said","name":"Charlotte"},{"index":0,"str":"Stefan and Roberto were subdued and even apprehensive on the day they joined Vlade and his friend Idelba on her tugboat","name":"Stefan"},{"index":3196,"str":"“I sure hope we find something today,” Stefan said","name":"Stefan"},{"index":3341,"str":"“Well,” Stefan said dubiously, “there is some doubt","name":"Stefan"},{"index":3781,"str":"“There’s no way our buoy will anchor this beast,” Stefan pointed out","name":"Stefan"},{"index":4741,"str":"“No way,” Stefan said","name":"Stefan"},{"index":4970,"str":"Vlade got Roberto and Stefan involved in moving the segments of the tube to the rear and latching them onto the long snake they were making","name":"Stefan"},{"index":6432,"str":"Roberto and Stefan grinned uncertainly","name":"Stefan"},{"index":7026,"str":"Roberto and Stefan glanced at each other and saw they both were feeling far out of their league but still in their element","name":"Stefan"},{"index":8083,"str":"Stefan and Roberto looked at each other wide-eyed","name":"Stefan"},{"index":8135,"str":"“That’s what we needed,” Stefan said","name":"Stefan"},{"index":13643,"str":"Stefan and Roberto danced around the adults, crawled between them, sniffed the dead stench of the wet muddy wood","name":"Stefan"},{"index":14470,"str":"It was great to see that the adults were just like Stefan and Roberto in this, that they still had that capacity in them even though grown up","name":"Stefan"},{"index":15049,"str":"Stefan and Roberto stared at each other with their Can you believe this? looks","name":"Stefan"},{"index":17606,"str":"“It’s right,” Stefan pointed out","name":"Stefan"},{"index":18075,"str":"Stefan and Roberto were appalled","name":"Stefan"},{"index":0,"str":"Stefan and Roberto were subdued and even apprehensive on the day they joined Vlade and his friend Idelba on her tugboat","name":"Roberto"},{"index":3042,"str":"“This has got to be the most powerful machine we have ever been on,” Roberto said","name":"Roberto"},{"index":3395,"str":"Roberto refused to accept this, shaking his head like a dog","name":"Roberto"},{"index":3851,"str":"“True,” Roberto said","name":"Roberto"},{"index":4302,"str":"“What if the anchor gets stuck down there?” Roberto asked Thabo","name":"Roberto"},{"index":4690,"str":"“Man, I wish I could go down again!” Roberto said","name":"Roberto"},{"index":4970,"str":"Vlade got Roberto and Stefan involved in moving the segments of the tube to the rear and latching them onto the long snake they were making","name":"Roberto"},{"index":6223,"str":"“I guess so,” Roberto said, struggling to comprehend the image","name":"Roberto"},{"index":6432,"str":"Roberto and Stefan grinned uncertainly","name":"Roberto"},{"index":7026,"str":"Roberto and Stefan glanced at each other and saw they both were feeling far out of their league but still in their element","name":"Roberto"},{"index":7474,"str":"Idelba used one of the nozzle’s hooks to lift the boys’ diving bell off the spot where Roberto had almost dug his own watery grave, as the old man put it","name":"Roberto"},{"index":7628,"str":"When that was placed well to the side, she returned the nozzle right to the red paint Roberto had put on the asphalt, which in the murky monochrome on the screens looked gray and ghostly, but that was okay, because now the nozzle’s hooks extended into the asphalt around the hole, and Thabo flipped a switch, and the grinding of the nozzle’s drill teeth cutting into the Bronx came out of their end of the tube with a sound they could hear in their guts","name":"Roberto"},{"index":8083,"str":"Stefan and Roberto looked at each other wide-eyed","name":"Roberto"},{"index":8173,"str":"“No lie,” Roberto said","name":"Roberto"},{"index":8382,"str":"Roberto grimaced and rubbed the screen of the radar as if that would clear the view of the bottom, now obscured by a flow of junk clouding the water","name":"Roberto"},{"index":13643,"str":"Stefan and Roberto danced around the adults, crawled between them, sniffed the dead stench of the wet muddy wood","name":"Roberto"},{"index":14470,"str":"It was great to see that the adults were just like Stefan and Roberto in this, that they still had that capacity in them even though grown up","name":"Roberto"},{"index":15049,"str":"Stefan and Roberto stared at each other with their Can you believe this? looks","name":"Roberto"},{"index":17372,"str":"“So whose are they?” Roberto asked, looking at Vlade","name":"Roberto"},{"index":17515,"str":"” Roberto did not have a poker face, and his crestfallen expression made the others laugh","name":"Roberto"},{"index":18075,"str":"Stefan and Roberto were appalled","name":"Roberto"},{"index":18109,"str":"“They’ll just take it away from us!” Roberto objected","name":"Roberto"},{"index":19930,"str":"Roberto was still focused on the gold, so now he interrupted them","name":"Roberto"}]},{"chapter":68,"name":"","length":412,"links":[]},{"chapter":69,"name":"Vlade","length":9194,"links":[{"index":0,"str":"The next day Vlade paid a visit to his friend Rosario O’Hara, one of the old veterans of the city’s subway squad","name":"Vlade"},{"index":113,"str":"In the years when Vlade had worked for her they had done all the usual subway work, which in those years included extending their operational reach into the drowned parts of the subway, slow work that mostly consisted of using the train tubes as giant water-filled utilidors, and laying within them things like conduit pipe for power lines, sewage pipes, tracks for robotic supply submersible capsules, comm cables, and so on, all the while keeping passage in them clear for maintenance access by city divers","name":"Vlade"},{"index":1088,"str":"Thus Vlade had spent ten years of his youth working for the LMMTA, and had strung a million miles of submersible line during that time, among other more interesting chores","name":"Vlade"},{"index":3016,"str":"Vlade had been worried about surveillance of the site, and Rosario had suggested they come at it from the side, as they might have done if it were one of their old work projects in the tunnels","name":"Vlade"},{"index":3210,"str":"Vlade liked that, and Trina and Jim did too; clearly they were all happy to have an excuse to do the stupid thing again","name":"Vlade"},{"index":3524,"str":"Vlade and Jim had worked together in the old days, and Vlade knew Jim was a great diver; it was good to see him again","name":"Vlade"},{"index":5290,"str":"Vlade swam toward the container and scoped it with an infrared scope they had brought along for this purpose","name":"Vlade"},{"index":6352,"str":"Rosario swam up to Vlade and spoke through their suits’ walkie-talkie system","name":"Vlade"},{"index":6961,"str":"Vlade finned hard to the side of the hot container","name":"Vlade"},{"index":7731,"str":"They asked the divers to go down and pull the inflatable staircase tube up to them, which Vlade and Jim did","name":"Vlade"},{"index":8347,"str":"After that Vlade and the others floated by the boat, waiting","name":"Vlade"},{"index":8662,"str":"This was something they could do that the police cruiser couldn’t (not optically, anyway), so after a while Vlade and Jim stayed down there by the container, looking around uneasily","name":"Vlade"}]},{"chapter":70,"name":"","length":51,"links":[]},{"chapter":71,"name":"Citizen","length":6657,"links":[]},{"chapter":72,"name":"","length":445,"links":[]},{"chapter":73,"name":"Inspector Gen","length":14052,"links":[{"index":1368,"str":"One of them, Ralph Muttchopf—brown hair thinning on top, about six foot, hound-dog face, skinny except for a slight pot belly—sat in a chair drinking coffee, looking around with a wary expression","name":"Mutt"},{"index":2226,"str":"Rosen had gotten sick at some point and Muttchopf had left messages on their food trays telling their captors about this, and meals after that had included some pills which Jeff had taken","name":"Mutt"},{"index":2667,"str":"Finally Muttchopf shook his head","name":"Mutt"},{"index":5891,"str":"There she found Muttchopf and Rosen ready to leave, and she and Olmstead escorted them onto the cruiser and headed down the East River toward home","name":"Mutt"},{"index":6545,"str":"Muttchopf nodded","name":"Mutt"},{"index":6660,"str":"“It smells like dinner,” Muttchopf declared","name":"Mutt"},{"index":7431,"str":"They nodded, shook their heads, shrugged, said little; then, with a look around, Muttchopf said to her, “How about you come up with us to our place when we’re done here","name":"Mutt"},{"index":8130,"str":"Muttchopf sat on the chair by his cot","name":"Mutt"},{"index":8427,"str":"Muttchopf nodded","name":"Mutt"},{"index":9548,"str":"Muttchopf spoke when it became clear Rosen wasn’t going to","name":"Mutt"},{"index":9962,"str":"Muttchopf had to tell the story","name":"Mutt"},{"index":10360,"str":"Then Jeff and Mutt had been fired","name":"Mutt"},{"index":10565,"str":"“He was cheating again,” Jeff added when Mutt had finished","name":"Mutt"},{"index":10738,"str":"Muttchopf’s lips were pulsing in and out as he regarded Gen, apparently assessing her level of financial acumen","name":"Mutt"},{"index":11666,"str":"“It’s hard to prove,” Muttchopf said","name":"Mutt"},{"index":12145,"str":"“We don’t know,” Muttchopf said","name":"Mutt"},{"index":1565,"str":"The other, Jeffrey Rosen—small, feral, triangular head covered with tight black curls—lay on an infirmary bed with an IV in his forearm","name":"Jeff"},{"index":2226,"str":"Rosen had gotten sick at some point and Muttchopf had left messages on their food trays telling their captors about this, and meals after that had included some pills which Jeff had taken","name":"Jeff"},{"index":2540,"str":"“How long were we in there?” Jeff asked","name":"Jeff"},{"index":2928,"str":"“That would be good,” Jeff said","name":"Jeff"},{"index":6602,"str":"“It’s cold,” Jeff added, shivering","name":"Jeff"},{"index":7645,"str":"Eventually they said they were stuffed, and Jeff was looking sleepy","name":"Jeff"},{"index":8444,"str":"“Look, the night we were snatched, Jeff here activated a covert channel he had inserted into one of the high-frequency trading cables of a company we’ve worked for a few times","name":"Jeff"},{"index":9635,"str":"And that was where Jeff put in his tap, in his cousin’s company’s dark pool diver","name":"Jeff"},{"index":10256,"str":"While doing the job Jeff had found evidence of malfeasance and taken it to his cousin; they had argued","name":"Jeff"},{"index":10360,"str":"Then Jeff and Mutt had been fired","name":"Jeff"},{"index":10565,"str":"“He was cheating again,” Jeff added when Mutt had finished","name":"Jeff"},{"index":10688,"str":"Jeff just shook his head, too disgusted to speak","name":"Jeff"},{"index":11369,"str":"“But illegal,” Jeff said, still disgusted","name":"Jeff"},{"index":11550,"str":"“I tried that before,” Jeff said","name":"Jeff"},{"index":11842,"str":"“I could prove it now,” Jeff muttered darkly","name":"Jeff"},{"index":12300,"str":"And Jeff had just tapped into the CME with his bite, and sent a package with the bitten-off points to the SEC","name":"Jeff"},{"index":0,"str":"Inspector Gen got a call from Vlade at around four that afternoon","name":"Gen"},{"index":446,"str":"“Really!” Gen said","name":"Gen"},{"index":935,"str":"Gen made arrangements for a water launch and asked Sergeant Olmstead to come with her","name":"Gen"},{"index":1811,"str":"Gen sat and inserted some questions into his nervous chatter","name":"Gen"},{"index":2581,"str":"Gen consulted her pad","name":"Gen"},{"index":2777,"str":"“I’m sure it did,” Gen said","name":"Gen"},{"index":2961,"str":"Gen left Olmstead there to guard them, warning the sergeant and the cops on duty there to take care; it was at least possible that the kidnappers had stuck trackers in them and might try to grab them back, or worse","name":"Gen"},{"index":3536,"str":"Gen walked into the fed building, got through security, and went to the office where the federal department of immigration, the FBI, the NYPD, and the Householders’ Union had combined to create a human smuggling task force","name":"Gen"},{"index":3899,"str":"Gen described the situation with her two rescued ones","name":"Gen"},{"index":5097,"str":"“Thanks,” Gen said","name":"Gen"},{"index":5212,"str":"“That might not be a bad thing,” Gen said","name":"Gen"},{"index":5594,"str":"” Gen finished her tea and rose to leave","name":"Gen"},{"index":5796,"str":"Gen thanked him and went back to her cruiser and headed back to the Frederick Douglass station","name":"Gen"},{"index":6040,"str":"The two men sat in chairs on the bridge beside Gen as she stood piloting, looking at the city like tourists","name":"Gen"},{"index":6365,"str":"“You must be kind of blown away,” Gen supposed","name":"Gen"},{"index":6734,"str":"“Low tide,” Gen pointed out","name":"Gen"},{"index":6908,"str":"At the Met they got off on the dock, and Gen had Olmstead run the cruiser back to the station","name":"Gen"},{"index":7002,"str":"Vlade greeted them, and he and Gen escorted the two men to the dining hall","name":"Gen"},{"index":7242,"str":"They heaped their plates high, and poured themselves glasses of the Flatiron’s red, and as they ate and drank, Gen sat across from them asking questions about the night of their abduction","name":"Gen"},{"index":8244,"str":"“Ah,” Gen said","name":"Gen"},{"index":8324,"str":"“So,” Gen said, “you seemed to be indicating that there was something you wanted to talk to me about?”","name":"Gen"},{"index":9367,"str":"Gen took this in","name":"Gen"},{"index":9900,"str":"Gen watched them closely","name":"Gen"},{"index":10657,"str":"“What do you mean?” Gen asked","name":"Gen"},{"index":10738,"str":"Muttchopf’s lips were pulsing in and out as he regarded Gen, apparently assessing her level of financial acumen","name":"Gen"},{"index":11351,"str":"“Nice,” Gen said","name":"Gen"},{"index":11495,"str":"“What if you had blown the whistle on him?” Gen asked","name":"Gen"},{"index":11888,"str":"“You could?” Gen said","name":"Gen"},{"index":12029,"str":"Gen stared at them","name":"Gen"},{"index":12434,"str":"Gen thought it over","name":"Gen"},{"index":12640,"str":"Next day Gen got a private pouch delivered to her by hand, from Goran’s office","name":"Gen"},{"index":13794,"str":"Gen pondered this","name":"Gen"},{"index":0,"str":"Inspector Gen got a call from Vlade at around four that afternoon","name":"Vlade"},{"index":7002,"str":"Vlade greeted them, and he and Gen escorted the two men to the dining hall","name":"Vlade"}]},{"chapter":74,"name":"","length":294,"links":[]},{"chapter":75,"name":"Charlotte","length":20081,"links":[{"index":6413,"str":"Mutt and Jeff nodded","name":"Mutt"},{"index":6537,"str":"Mutt and Jeff shrugged","name":"Mutt"},{"index":9697,"str":"“Just ask Mutt and Jeff here","name":"Mutt"},{"index":9729,"str":"“We were held hostage in a secret location,” Mutt explained to the mystified Amelia","name":"Mutt"},{"index":15008,"str":"Hexter cross-eyed, Mutt and Jeff not yet returned to this planet, Amelia busy leaving it by way of the wine","name":"Mutt"},{"index":18485,"str":"“What do you think?” Charlotte asked Mutt and Jeff","name":"Mutt"},{"index":18537,"str":"Mutt waggled a hand","name":"Mutt"},{"index":18717,"str":"Jeff and Mutt looked at each other","name":"Mutt"},{"index":18914,"str":"“The system,” Mutt suggested","name":"Mutt"},{"index":19285,"str":"Mutt regarded her","name":"Mutt"},{"index":19767,"str":"“Please,” Mutt said","name":"Mutt"},{"index":20006,"str":"When they all had gotten refills they toasted Mutt and Jeff’s safe return","name":"Mutt"},{"index":6413,"str":"Mutt and Jeff nodded","name":"Jeff"},{"index":6537,"str":"Mutt and Jeff shrugged","name":"Jeff"},{"index":6994,"str":"“We persist in living,” Jeff said sardonically","name":"Jeff"},{"index":9697,"str":"“Just ask Mutt and Jeff here","name":"Jeff"},{"index":9830,"str":"“I almost died,” Jeff said","name":"Jeff"},{"index":15008,"str":"Hexter cross-eyed, Mutt and Jeff not yet returned to this planet, Amelia busy leaving it by way of the wine","name":"Jeff"},{"index":16937,"str":"“Four billion dollars is just the start of it,” Jeff muttered darkly","name":"Jeff"},{"index":18485,"str":"“What do you think?” Charlotte asked Mutt and Jeff","name":"Jeff"},{"index":18651,"str":"“And staying out of their clutches,” Jeff muttered","name":"Jeff"},{"index":18717,"str":"Jeff and Mutt looked at each other","name":"Jeff"},{"index":18944,"str":"“Capital,” Jeff clarified","name":"Jeff"},{"index":19404,"str":"“Nice work if you can get it!” Jeff pointed out crabbily","name":"Jeff"},{"index":19788,"str":"Jeff scowled","name":"Jeff"},{"index":20006,"str":"When they all had gotten refills they toasted Mutt and Jeff’s safe return","name":"Jeff"},{"index":6875,"str":"Gen took care of all that","name":"Gen"},{"index":14382,"str":"I’d be very interested to hear what Inspector Gen would say about it","name":"Gen"},{"index":14645,"str":"On the other hand, Inspector Gen was a resident and a known presence","name":"Gen"},{"index":15117,"str":"Charlotte pinged Gen, found she was down in her room","name":"Gen"},{"index":15171,"str":"“Gen, could you come up to the farm and give us an opinion on a city issue?”","name":"Gen"},{"index":15249,"str":"A few minutes later, Inspector Gen Octaviasdottir was standing there before them, tall and massive in the dark, hard to see well","name":"Gen"},{"index":15547,"str":"Gen watched them politely as they spoke","name":"Gen"},{"index":15685,"str":"Gen continued to look at them, blinking as she regarded them each in turn","name":"Gen"},{"index":15814,"str":"Gen shrugged","name":"Gen"},{"index":16240,"str":"Gen shrugged","name":"Gen"},{"index":18040,"str":"Gen was shaking her head and helping Amelia open a second bottle","name":"Gen"},{"index":18269,"str":"But there was their Inspector Gen, famous policewoman, a power in the city, a pillar of the SuperVenice, happily ignoring this bad fate by conferring with Amelia about vintages of vinho verde or some such nonsense","name":"Gen"},{"index":17593,"str":"“Franklin Garr? I like him","name":"Franklin"},{"index":4820,"str":"After she had spoken with a number of people who came up to commiserate or argue, Vlade approached","name":"Vlade"},{"index":4919,"str":"It was clear he wanted to talk to her in private, so she made her excuses to the last clutch of residents, who would have been happy to argue all night long, and followed Vlade to the elevators","name":"Vlade"},{"index":5159,"str":"“Some things have come up you should know about,” Vlade said","name":"Vlade"},{"index":5856,"str":"Vlade led the way over to the hotellos and called out, “Knock knock, we’re here to visit","name":"Vlade"},{"index":5976,"str":"“Too crowded in there,” Vlade replied","name":"Vlade"},{"index":6139,"str":"Out of the tent emerged the two boys Vlade indulged around the docks, and the old man they had befriended and then rescued from his drowned squat; and then the two men who had disappeared from the farm so many weeks ago","name":"Vlade"},{"index":6561,"str":"Vlade said, “We were over in the Bronx doing some treasure hunting with the boys here, and we found these guys in a container down in the Cypress subway hole","name":"Vlade"},{"index":6771,"str":"“Yeah,” Vlade said","name":"Vlade"},{"index":7230,"str":"“Come on!” Vlade objected","name":"Vlade"},{"index":7393,"str":"They sat around the vegetable cleaning and cutting table, and Vlade uncorked the bottle and poured wine into white ceramic coffee cups","name":"Vlade"},{"index":7928,"str":"” Holding his cup out to Vlade","name":"Vlade"},{"index":7960,"str":"“Quit it,” Vlade said","name":"Vlade"},{"index":7983,"str":"Then while the two men were telling Charlotte their tale, Vlade went to the elevator and came back with Amelia Black","name":"Vlade"},{"index":8611,"str":"“We heard,” Vlade said","name":"Vlade"},{"index":8790,"str":"“We heard,” Vlade repeated","name":"Vlade"},{"index":9962,"str":"“And so are you,” Vlade reminded her","name":"Vlade"},{"index":10677,"str":"“It’s a good idea,” Vlade said in transition, “but for now, the boys and Mr","name":"Vlade"},{"index":10840,"str":"She knew Vlade was very fond of their resident cloud star, but to Charlotte she seemed just as spacey and superficial as she did on her program, not that Charlotte had ever watched more than ten minutes of it","name":"Vlade"},{"index":11313,"str":"“But, on the other hand,” Vlade weighed in heavily, “we didn’t take the offer","name":"Vlade"},{"index":11774,"str":"And the boys had dived it using their own diving bell (“Wait, what?” Charlotte said), and there it was, right where predicted, but down under twenty feet of mud and landfill, an unwieldy goo, impossible for the boys to dig up on their own, so Vlade had enlisted the help of his friends Idelba and Thabo, who ran a huge, huge, gigantic sand dredge out at Coney Island, they were moving Coney Island’s beach up to the new shoreline twenty blocks north, and for them digging up the Hussar’s treasure chest (actual treasure chests, small but insanely heavy) was nothing, it was toothpick work, and now Idelba and Thabo were part of their consortium, joining the people right here around this table","name":"Vlade"},{"index":13093,"str":"“We didn’t tell anyone,” Vlade clarified","name":"Vlade"},{"index":13632,"str":"“They found a cannon of the Hussar a long time ago,” Vlade said","name":"Vlade"},{"index":14005,"str":"“Well,” Vlade said, “look at it this way","name":"Vlade"},{"index":14875,"str":"“Would she?” Vlade asked","name":"Vlade"},{"index":14915,"str":"Vlade shrugged, looked around at the others","name":"Vlade"},{"index":15378,"str":"They invited her to sit down, and then hesitantly, as if it were some kind of hypothetical case, Vlade and Mr","name":"Vlade"},{"index":17204,"str":"“Sounds dangerous,” Vlade said","name":"Vlade"},{"index":17441,"str":"“There’s finance people in the building,” Vlade said","name":"Vlade"},{"index":17623,"str":"Vlade rolled his eyes at her just like Larry used to back in the day","name":"Vlade"},{"index":5278,"str":"Most of the people involved are up there, and Amelia is just about to arrive and tie off her blimp, and she might be a good one to have in on this too","name":"Amelia"},{"index":7983,"str":"Then while the two men were telling Charlotte their tale, Vlade went to the elevator and came back with Amelia Black","name":"Amelia"},{"index":8184,"str":"“Amelia’s back,” he said unnecessarily, and made introductions all around","name":"Amelia"},{"index":8258,"str":"Charlotte had only met the cloud star once before, and was content to be introduced again, as Amelia didn’t seem to recall their earlier meeting, in their conversation over the phone when Amelia had been trapped in her blimp’s closet","name":"Amelia"},{"index":8540,"str":"“Well I’m not,” Amelia said, tearing up again","name":"Amelia"},{"index":8666,"str":"Amelia gave her a bereft look and said, “I just mean I was the one who took them down to Antarctica","name":"Amelia"},{"index":8818,"str":"“Fucking Antarctic Defense League,” Amelia said","name":"Amelia"},{"index":9071,"str":"Amelia was scowling","name":"Amelia"},{"index":9422,"str":"“I didn’t!” Amelia protested","name":"Amelia"},{"index":9729,"str":"“We were held hostage in a secret location,” Mutt explained to the mystified Amelia","name":"Amelia"},{"index":9858,"str":"“I’m sorry,” Amelia said","name":"Amelia"},{"index":10184,"str":"“Not my polar bears,” Amelia objected","name":"Amelia"},{"index":10467,"str":"Protecting endangered species in secret was a paradigm buster that left Amelia obviously conflicted, or even confused","name":"Amelia"},{"index":11247,"str":"“Did that happen?” Amelia cried","name":"Amelia"},{"index":12470,"str":"“Gold?” Charlotte and Amelia said together","name":"Amelia"},{"index":15008,"str":"Hexter cross-eyed, Mutt and Jeff not yet returned to this planet, Amelia busy leaving it by way of the wine","name":"Amelia"},{"index":16346,"str":"” She gestured to Amelia to pour her a cup of wine","name":"Amelia"},{"index":18040,"str":"Gen was shaking her head and helping Amelia open a second bottle","name":"Amelia"},{"index":18269,"str":"But there was their Inspector Gen, famous policewoman, a power in the city, a pillar of the SuperVenice, happily ignoring this bad fate by conferring with Amelia about vintages of vinho verde or some such nonsense","name":"Amelia"},{"index":19172,"str":"“Me too,” Amelia interjected","name":"Amelia"},{"index":19934,"str":"Amelia smiled the smile that had made her a cloud star, filled her cup","name":"Amelia"},{"index":132,"str":"Charlotte’s half-assed investigations had not been able to crack the façade there, and in any case, no matter who was behind it, the CC&Rs of the co-op required that a vote be taken on matters like this within ninety days of their initiation, and this was day eighty-nine, and she wanted no technical infractions to cause trouble later","name":"Charlotte"},{"index":1086,"str":"Charlotte gazed at the fellow citizens of their little city-state with such trepidation and political distrust that really it seemed like a new kind of fear","name":"Charlotte"},{"index":1993,"str":"Charlotte couldn’t stay focused long enough to catch more than the pro or con expressed, leaving the gist of people’s arguments to some later time when she might give a shit","name":"Charlotte"},{"index":2641,"str":"Finally Mariolino looked up at Charlotte and then the others in the room","name":"Charlotte"},{"index":2904,"str":"Charlotte was both relieved and worried","name":"Charlotte"},{"index":6360,"str":"“Hey!” Charlotte said to the two men","name":"Charlotte"},{"index":6721,"str":"Charlotte was amazed","name":"Charlotte"},{"index":7042,"str":"“Good idea,” Charlotte said, and sat down heavily on a chair by the railing","name":"Charlotte"},{"index":7983,"str":"Then while the two men were telling Charlotte their tale, Vlade went to the elevator and came back with Amelia Black","name":"Charlotte"},{"index":8258,"str":"Charlotte had only met the cloud star once before, and was content to be introduced again, as Amelia didn’t seem to recall their earlier meeting, in their conversation over the phone when Amelia had been trapped in her blimp’s closet","name":"Charlotte"},{"index":8494,"str":"“We’re celebrating,” Charlotte said grumpily","name":"Charlotte"},{"index":8635,"str":"“Your bears?” Charlotte asked","name":"Charlotte"},{"index":8922,"str":"“That’s what they like about it,” Charlotte supposed sourly","name":"Charlotte"},{"index":9310,"str":"“You could always move them in secret,” Charlotte suggested","name":"Charlotte"},{"index":9541,"str":"Besides, do you really think anything happens in secret anymore?” she asked, as if Charlotte were naïve","name":"Charlotte"},{"index":9647,"str":"“Lots of things happen in secret,” Charlotte said","name":"Charlotte"},{"index":10401,"str":"“Just do it again,” Charlotte suggested again","name":"Charlotte"},{"index":10788,"str":"Charlotte nodded, relieved at the change of subject","name":"Charlotte"},{"index":10840,"str":"She knew Vlade was very fond of their resident cloud star, but to Charlotte she seemed just as spacey and superficial as she did on her program, not that Charlotte had ever watched more than ten minutes of it","name":"Charlotte"},{"index":11280,"str":"“It did,” Charlotte said grimly","name":"Charlotte"},{"index":11496,"str":"“Which means what?” Charlotte asked","name":"Charlotte"},{"index":11774,"str":"And the boys had dived it using their own diving bell (“Wait, what?” Charlotte said), and there it was, right where predicted, but down under twenty feet of mud and landfill, an unwieldy goo, impossible for the boys to dig up on their own, so Vlade had enlisted the help of his friends Idelba and Thabo, who ran a huge, huge, gigantic sand dredge out at Coney Island, they were moving Coney Island’s beach up to the new shoreline twenty blocks north, and for them digging up the Hussar’s treasure chest (actual treasure chests, small but insanely heavy) was nothing, it was toothpick work, and now Idelba and Thabo were part of their consortium, joining the people right here around this table","name":"Charlotte"},{"index":12470,"str":"“Gold?” Charlotte and Amelia said together","name":"Charlotte"},{"index":12846,"str":"“Aren’t there laws about salvaging sunken treasure?” Charlotte asked","name":"Charlotte"},{"index":13053,"str":"“You ignored the laws,” Charlotte said","name":"Charlotte"},{"index":13951,"str":"Charlotte shook her head","name":"Charlotte"},{"index":14161,"str":"“Hmm,” Charlotte said","name":"Charlotte"},{"index":14305,"str":"” Charlotte thought it over","name":"Charlotte"},{"index":14816,"str":"“Come on,” Charlotte said","name":"Charlotte"},{"index":15117,"str":"Charlotte pinged Gen, found she was down in her room","name":"Charlotte"},{"index":15589,"str":"“So,” Charlotte said at the end of their recitation, “what do you think we should do about it?”","name":"Charlotte"},{"index":15889,"str":"Charlotte stared at her","name":"Charlotte"},{"index":15967,"str":"” Slightly slow and pointed with that last sentence, and including a glance at Charlotte","name":"Charlotte"},{"index":16057,"str":"“Sorry,” Charlotte said","name":"Charlotte"},{"index":16576,"str":"Charlotte, you’re a lawyer, you know what I’m saying","name":"Charlotte"},{"index":16748,"str":"“We already own the building,” Charlotte complained, still aggrieved by the night’s vote","name":"Charlotte"},{"index":17007,"str":"“What do you mean?” Charlotte asked","name":"Charlotte"},{"index":17269,"str":"“I hate that kind of thing,” Charlotte said","name":"Charlotte"},{"index":17575,"str":"Charlotte frowned","name":"Charlotte"},{"index":17888,"str":"“That would be interesting,” Charlotte allowed","name":"Charlotte"},{"index":18106,"str":"Charlotte sighed and gave up on that issue","name":"Charlotte"},{"index":18485,"str":"“What do you think?” Charlotte asked Mutt and Jeff","name":"Charlotte"},{"index":18752,"str":"They were like feral twins at this point, Charlotte thought","name":"Charlotte"},{"index":19017,"str":"“Not my brain,” Charlotte declared","name":"Charlotte"},{"index":19112,"str":"“I hate that shit,” Charlotte said","name":"Charlotte"},{"index":19231,"str":"“I want it for this building,” Charlotte said grimly","name":"Charlotte"},{"index":19461,"str":"Charlotte glared at him, and he raised a hand to ward her off: “Hey, I like the concept! It’s just not that easy","name":"Charlotte"},{"index":19640,"str":"“But did you really try?” Charlotte inquired","name":"Charlotte"},{"index":19862,"str":"” Charlotte looked around at them, stuck out her coffee cup for seconds","name":"Charlotte"},{"index":7362,"str":"“Big news,” Roberto declared","name":"Roberto"},{"index":7765,"str":"Roberto snorted at this and downed his shot like an Italian espresso","name":"Roberto"},{"index":9883,"str":"She drained her cup in a single swallow, like Roberto","name":"Roberto"},{"index":13498,"str":"“The ship was smooshed,” Roberto said","name":"Roberto"},{"index":16187,"str":"“What about the law?” Roberto said","name":"Roberto"}]},{"chapter":76,"name":"","length":742,"links":[]},{"chapter":77,"name":"Franklin","length":16418,"links":[{"index":13906,"str":"They nationalized General Motors, a car company, and kept it running until it was back on its feet and paid off its debt to the people","name":"Gen"},{"index":0,"str":"So the building super, Vlade the derailer, came over one morning when he was pulling my bug out of the rafters of his ever-more-crowded boathouse, leering in what appeared to be his attempt at a friendly smile","name":"Vlade"},{"index":3020,"str":"After a perfunctory dinner in the commons I went up to the farm floor and found Charlotte already there, with Vlade and the old man that the two boys had taken in, and Amelia Black the cloud babe, plus a couple of men who looked like hobos","name":"Vlade"},{"index":3366,"str":"“What’s up?” I said, taking a coffee cup of wine from Vlade","name":"Vlade"},{"index":4189,"str":"Meaning Vlade, me, and Mr","name":"Vlade"},{"index":4224,"str":"And a couple of Vlade’s friends","name":"Vlade"},{"index":2652,"str":"I turned into Twenty-third and hummed to the Met, which was still flying Amelia Black’s blimp like a big wind sock, the late sun burnishing the gilded cupola under it","name":"Amelia"},{"index":3020,"str":"After a perfunctory dinner in the commons I went up to the farm floor and found Charlotte already there, with Vlade and the old man that the two boys had taken in, and Amelia Black the cloud babe, plus a couple of men who looked like hobos","name":"Amelia"},{"index":3619,"str":"Amelia Black kept the wine bottle on the floor by her chair","name":"Amelia"},{"index":7450,"str":"“You look like you are,” Amelia Black tossed in","name":"Amelia"},{"index":7841,"str":"Charlotte aimed a look at Amelia, like, Don’t encourage him","name":"Amelia"},{"index":10044,"str":"“I like that,” Amelia Black said","name":"Amelia"},{"index":11063,"str":"Amelia was laughing to see the others laugh","name":"Amelia"},{"index":15489,"str":"“But wouldn’t people get in trouble?” Amelia inquired","name":"Amelia"},{"index":15933,"str":"“I like this!” Amelia exclaimed","name":"Amelia"},{"index":16166,"str":"“Me too!” Amelia said","name":"Amelia"},{"index":462,"str":"“Charlotte wants to talk to you,” he said","name":"Charlotte"},{"index":511,"str":"“So you want to talk to Charlotte","name":"Charlotte"},{"index":876,"str":"Turns out they are needing some investment advice, and Charlotte and I are stepping in as their help","name":"Charlotte"},{"index":1100,"str":"“Charlotte will explain the situation","name":"Charlotte"},{"index":3020,"str":"After a perfunctory dinner in the commons I went up to the farm floor and found Charlotte already there, with Vlade and the old man that the two boys had taken in, and Amelia Black the cloud babe, plus a couple of men who looked like hobos","name":"Charlotte"},{"index":3427,"str":"Charlotte clinked her coffee cup with mine","name":"Charlotte"},{"index":3553,"str":"I sat down with Charlotte facing me, and the others sat around us","name":"Charlotte"},{"index":3681,"str":"Charlotte said, “Our boys, Roberto and Stefan, have inherited some money","name":"Charlotte"},{"index":3861,"str":"“Anything’s possible,” Charlotte said, then frowned, as if realizing the inaccuracy of that statement","name":"Charlotte"},{"index":4079,"str":"“They’re like brothers,” Charlotte said","name":"Charlotte"},{"index":4602,"str":"“The point is,” Charlotte said, still smiling, “they want to help the co-op, and they know you and trust you","name":"Charlotte"},{"index":5020,"str":"“Joke,” Charlotte reassured me","name":"Charlotte"},{"index":5479,"str":"“Okay,” Charlotte said","name":"Charlotte"},{"index":7213,"str":"Charlotte was nodding like she knew this already","name":"Charlotte"},{"index":7841,"str":"Charlotte aimed a look at Amelia, like, Don’t encourage him","name":"Charlotte"},{"index":8712,"str":"“Fictional money?” Charlotte asked him","name":"Charlotte"},{"index":9125,"str":"“Stop it,” Charlotte said","name":"Charlotte"},{"index":9939,"str":"Charlotte stared at me as if trying to comprehend some kind of mystery","name":"Charlotte"},{"index":10078,"str":"Charlotte shook her head hard: Quit encouraging him! “Any other methods you can suggest?”","name":"Charlotte"},{"index":10543,"str":"Charlotte frowned","name":"Charlotte"},{"index":11148,"str":"As did Charlotte, now that I finally saw it","name":"Charlotte"},{"index":11194,"str":"“Tell me how,” Charlotte said, eyes alight with the notion of destroying civilization","name":"Charlotte"},{"index":12246,"str":"Charlotte was regarding me with a laser eye","name":"Charlotte"},{"index":12647,"str":"Charlotte said, “That’s what I do","name":"Charlotte"},{"index":14624,"str":"Charlotte looked at the two prodigal quants","name":"Charlotte"},{"index":14755,"str":"Charlotte palmed her forehead","name":"Charlotte"},{"index":15817,"str":"Now Charlotte was smiling at me again, eyes alight, really an intelligent and warm smile","name":"Charlotte"},{"index":16189,"str":"Charlotte laughed at them","name":"Charlotte"},{"index":3681,"str":"Charlotte said, “Our boys, Roberto and Stefan, have inherited some money","name":"Stefan"},{"index":3681,"str":"Charlotte said, “Our boys, Roberto and Stefan, have inherited some money","name":"Roberto"}]},{"chapter":78,"name":"","length":1008,"links":[]},{"chapter":79,"name":"Amelia","length":32437,"links":[{"index":20578,"str":"She searched her wristpad and brought up a recording of her undergraduate advisor at the University of Wisconsin, an evolution and ecology theorist named Lucky Jeff, whose voice even now had the power to soothe her","name":"Jeff"},{"index":86,"str":"At first she liked Franklin from the building, a good-looking man, but he thought she was a simpleton, so then she didn’t like him","name":"Franklin"},{"index":0,"str":"Amelia banged around New York for a few days, too angry and distracted to do anything","name":"Amelia"},{"index":594,"str":"“Amelia, you’ve got to stop with that,” Nicole said","name":"Amelia"},{"index":2564,"str":"Nicole was acting weird, like Amelia was some kind of bomb that might go off","name":"Amelia"},{"index":2852,"str":"In her childhood Amelia had learned to skate on ponds and rivers, so she could handle the canals’ rough patches, and skate backward, which was fun, and even twirl a little, although this was not so fun, as it reminded her of when her mom had made her do things for contests","name":"Amelia"},{"index":3127,"str":"Her mom had been a stage mom, and Amelia supposed she had to be grateful now that she was a performing artist, but she wasn’t","name":"Amelia"},{"index":4128,"str":"Amelia skated at speed, going faster than the taxis of earlier times would have been able to, because she could dodge through traffic like a motorcyclist","name":"Amelia"},{"index":7123,"str":"Everyone but Amelia laughed at that, although she would have too if she could have, because it had of course been a major feature of her shows during those first couple years in the cloud","name":"Amelia"},{"index":7872,"str":"Nicole was just outside the shower door, keeping dry but also checking her out and Amelia supposed filming her","name":"Amelia"},{"index":9128,"str":"“You are all so nice!” Amelia said","name":"Amelia"},{"index":9258,"str":"When Amelia had recovered from her dunking in the canal, she got in the Assisted Migration and flew from New York to the northeastern coast of Greenland","name":"Amelia"},{"index":10208,"str":"In any case, Amelia had visited New Copenhagen before and was pleased when Frans guided the Assisted Migration down to the long line of masts at the southern end of the city, where a short fjord cut north to the island’s center, giving the island and city the shape of a horseshoe","name":"Amelia"},{"index":11129,"str":"When her airship was secured, Amelia took a bus to the head of the fjord, where the biggest pedestrian district was located","name":"Amelia"},{"index":11552,"str":"It reminded Amelia of Boston","name":"Amelia"},{"index":11677,"str":"Amelia’s local friends from the Wildlife Migrators Association had gathered there to commiserate with her over her disastrous voyage south, to drink the memory away, and to discuss new plans","name":"Amelia"},{"index":12306,"str":"“I know that,” Amelia said sulkily","name":"Amelia"},{"index":12781,"str":"“I hate that shit,” Amelia said","name":"Amelia"},{"index":13010,"str":"He stared intently at her, and despite his rebukes, Amelia began to get the idea that he was coming on to her","name":"Amelia"},{"index":13820,"str":"“Maybe if we made a deal with them,” Amelia said","name":"Amelia"},{"index":14030,"str":"Amelia sighed gloomily","name":"Amelia"},{"index":14224,"str":"“It’s too warm up here now,” Amelia said","name":"Amelia"},{"index":15062,"str":"Amelia found this depressing","name":"Amelia"},{"index":15820,"str":"They interacted with some fantasy Amelia in their heads, a mix of her show’s persona and her actual presence, and she played to that, and it made things easier in some ways","name":"Amelia"},{"index":17462,"str":"Amelia sighed","name":"Amelia"},{"index":19861,"str":"So Amelia passed these days looking down at the world, and as she did she tried to think things through","name":"Amelia"},{"index":22964,"str":"As Amelia listened to this, wondering if any of it was true, or if it could help her if it were, she was looking down at Kamchatka","name":"Amelia"},{"index":24715,"str":"Amelia loved skyvillages","name":"Amelia"},{"index":25238,"str":"People spoke very highly of life in these villages, and Amelia always enjoyed her visits to them","name":"Amelia"},{"index":25527,"str":"Amelia was now flying at around ten thousand feet, so the skyvillages she saw below her looked like flower arrangements, or cloisonné jewelry","name":"Amelia"},{"index":27548,"str":"Amelia took the controls and headed down after the distressed aircraft","name":"Amelia"},{"index":27665,"str":"Amelia piloted, and Frans took over propulsion and ballast, and also established contact with the skyvillage, which was now putting out a mayday","name":"Amelia"},{"index":28208,"str":"After her recent adventure putting the Assisted Migration on the vertical to deal with the bears, Amelia could well imagine the chaos","name":"Amelia"},{"index":28910,"str":"The Assisted Migration dropped toward the sinking skyvillage, much more slowly than Amelia would have liked, even in what seemed to her some kind of slow motion, but in fact they were dropping fast, Frans said","name":"Amelia"},{"index":29558,"str":"When they reached an altitude just above the falling skyvillage and had lowered Amelia’s swing rope with a grappling hook on its end, people on the skyvillage ventured out onto their sharply canted platform, all of them roped and harnessed like climbers, to collect the Assisted Migration’s grapple and hook it to the edge of the village floor, midway around the arc of busted balloons","name":"Amelia"},{"index":29944,"str":"It was so amazing to see the villagers out there in their harnesses, maneuvering like mountain climbers, that Amelia started to film it","name":"Amelia"},{"index":30082,"str":"“Hey people,” she said to the cloud, “this is Amelia, I’m back","name":"Amelia"},{"index":31972,"str":"“And make a garden wilder than the wild,” Amelia sang, the last line of her show’s theme song, from the great poem by Frederick Turner","name":"Amelia"}]},{"chapter":80,"name":"","length":318,"links":[]},{"chapter":81,"name":"Inspector Gen","length":14173,"links":[{"index":2106,"str":"In this Chinese archive the Lame Ass team had been able to locate the very container in which Mutt and Jeff had been imprisoned","name":"Mutt"},{"index":4017,"str":"What the analysts had recovered before that move showed nothing at all concerning the kidnapping of Rosen and Muttchopf, but they had found evidence of contacts with another sucker on that leg of the octopus, a group implicated in three corporate assassinations","name":"Mutt"},{"index":4418,"str":"Rosen and Muttchopf’s names could be in some of these data, but if they had been given code names that hadn’t been figured out, that might explain why they hadn’t appeared on any of these lists","name":"Mutt"},{"index":4890,"str":"Here there was a record of the hiring of Rosen and Muttchopf, also of a contact with Pinscher for personal security consultations","name":"Mutt"},{"index":2106,"str":"In this Chinese archive the Lame Ass team had been able to locate the very container in which Mutt and Jeff had been imprisoned","name":"Jeff"},{"index":5059,"str":"The Lame Ass analysts had also snatched some dark pool diving algorithms out of the dark pools themselves; these had been tagged by Jeff Rosen as being his work, and they stuck to other algorithms he had spotted in the dark pools","name":"Jeff"},{"index":5808,"str":"If Jeff had been seeing only the tip of the iceberg, in terms of illegal market manipulation, sequestering him and his partner could have saved Vinson from years in prison, or at least an inconvenient slap on the wrist","name":"Jeff"},{"index":9547,"str":"“I’m here to ask you about your cousin Jeff Rosen,” Gen said","name":"Jeff"},{"index":10390,"str":"“When did you last see your cousin, Jeff Rosen?”","name":"Jeff"},{"index":0,"str":"Inspector Gen and Sergeant Olmstead went to talk to the Lower Manhattan Mutual Aid Society’s data analysis team, a group of quanty detectives who were always striving to mine the stacks and the cloud in ways cleverer than the official city and federal teams","name":"Gen"},{"index":896,"str":"Gen followed Olmstead through their security with the gloomy sensation she always had when entering this bastion of big data","name":"Gen"},{"index":1022,"str":"To her data analysis was the ugly love child of science and Kafka, always either proving the sky was blue or demonstrating the truth of something deeply wrong or, to be more precise, radically counterintuitive to Gen Octaviasdottir","name":"Gen"},{"index":1255,"str":"And Gen was all about intuition","name":"Gen"},{"index":4781,"str":"“Damn,” Gen said","name":"Gen"},{"index":5532,"str":"Gen pondered her options now by running various scenarios past Sean Olmstead, who served as her whiteboard in the absence of a real one","name":"Gen"},{"index":7246,"str":"Gen sighed","name":"Gen"},{"index":8270,"str":"It made Gen happy to stroll purposefully along, as if marshaling a tiny parade","name":"Gen"},{"index":8435,"str":"“We’re here to speak to Henry Vinson, at Alban Albany,” Gen said to the building security people","name":"Gen"},{"index":8596,"str":"Gen chewed vigorously to pop her ears on the way up to the fiftieth floor, which was fairly low in the tower, where the floors were largest","name":"Gen"},{"index":8906,"str":"“I want to speak to Henry Vinson,” Gen said, showing them the warrant","name":"Gen"},{"index":8977,"str":"One of the receptionists gestured at her phone and Gen said, “Yes, go ahead,” and she pinged Vinson and said that there was a policewoman to see him","name":"Gen"},{"index":9413,"str":"Like an actor playing a chief executive officer, but this, Gen found, was almost always true of CEOs","name":"Gen"},{"index":9547,"str":"“I’m here to ask you about your cousin Jeff Rosen,” Gen said","name":"Gen"},{"index":10176,"str":"“No,” Inspector Gen agreed","name":"Gen"},{"index":11380,"str":"Gen paused a beat, to let that remark reverberate a little","name":"Gen"},{"index":12160,"str":"“No,” Gen said","name":"Gen"},{"index":12722,"str":"“You’re kidding,” Gen said","name":"Gen"},{"index":12842,"str":"Gen pondered","name":"Gen"},{"index":13019,"str":"Gen looked at Lieutenant Claire, who shrugged; nothing to be done","name":"Gen"},{"index":13085,"str":"Gen said, “We are leaving under protest, registered here and now","name":"Gen"},{"index":13419,"str":"When they were on the police boat, Gen said, “Those fuckers","name":"Gen"},{"index":13683,"str":"“Good work,” Gen said to Claire","name":"Gen"},{"index":14023,"str":"Gen wondered if that would be the only good result she would get out of this move","name":"Gen"}]},{"chapter":82,"name":"","length":525,"links":[]},{"chapter":83,"name":"Citizen","length":12409,"links":[]},{"chapter":84,"name":"","length":229,"links":[]},{"chapter":85,"name":"Stefan and Roberto","length":23863,"links":[{"index":11504,"str":"Even Franklin Garr’s zoomer couldn’t have beat it","name":"Franklin"},{"index":0,"str":"Roberto and Stefan loved it when the great harbor froze over","name":"Stefan"},{"index":495,"str":"The boys looked at each other uncertainly, but Stefan said, “Sure, Mr","name":"Stefan"},{"index":1216,"str":"“We scavenge,” Stefan said","name":"Stefan"},{"index":1761,"str":"“But how do you attach them so you can steer?” Stefan asked","name":"Stefan"},{"index":2867,"str":"“That’s why they call them that,” Stefan supposed","name":"Stefan"},{"index":6137,"str":"“Like Melville after his readers left,” Stefan observed","name":"Stefan"},{"index":7073,"str":"“But that underwater digging turned out to be hard,” Stefan pointed out","name":"Stefan"},{"index":7513,"str":"Stefan looked unconvinced","name":"Stefan"},{"index":8154,"str":"Stefan tugged on the two lines running up to the rudder post until the front skate pointed a little to the right, up into the wind, then wrapped those lines to their own cleats","name":"Stefan"},{"index":8616,"str":"Stefan and Roberto looked at each other round-eyed, and they might even have been nervous if Mr","name":"Stefan"},{"index":8884,"str":"So Roberto kept the sail taut, and Stefan pulled the front skate a little more to the right, pointing them up into the wind a bit more, and they clatterswooshed across the mighty river, which from this vantage appeared like an immense ice lake, like one of the Great Lakes maybe","name":"Stefan"},{"index":9529,"str":"Stefan uncleated his lines and pulled hard on the right one, which made the boat curve to the right, upriver and upwind, until the sail fluttered hard, and they scratched over the ice to a halt, with only the flapping sail moving","name":"Stefan"},{"index":10352,"str":"“We should have thought of it before,” Stefan said","name":"Stefan"},{"index":10765,"str":"Stefan pulled on his lines, twisting the front skate left to get them to turn the front back toward the city as they were being shoved backward","name":"Stefan"},{"index":11200,"str":"Roberto hauled with all his might on the sheet and cleated it down before he lost it, and Stefan lay down to stay under the boom, which was now angled over the right side of the boat instead of the left","name":"Stefan"},{"index":13831,"str":"“What spot could be out here?” Stefan wondered","name":"Stefan"},{"index":14586,"str":"Stefan and Roberto stared at him","name":"Stefan"},{"index":15300,"str":"“What’s a smuggled good?” Stefan clarified, glancing at Roberto","name":"Stefan"},{"index":19027,"str":"“So then what?” Stefan insisted","name":"Stefan"},{"index":21692,"str":"“Yikes,” Stefan said","name":"Stefan"},{"index":23068,"str":"“You’re kidding,” Stefan said","name":"Stefan"},{"index":0,"str":"Roberto and Stefan loved it when the great harbor froze over","name":"Roberto"},{"index":2062,"str":"Roberto said, “It bugs me that it isn’t being put in a museum or something","name":"Roberto"},{"index":2435,"str":"“That would be a blocknecklace for sure,” Roberto said","name":"Roberto"},{"index":4424,"str":"“Will that work?” Roberto asked","name":"Roberto"},{"index":4825,"str":"Hexter?” Roberto asked as they worked","name":"Roberto"},{"index":6732,"str":"“A manuscript isn’t going to hold up underwater like gold,” Roberto pointed out","name":"Roberto"},{"index":7481,"str":"“Great idea,” Roberto enthused","name":"Roberto"},{"index":7718,"str":"“Let’s give it a try!” Roberto cried","name":"Roberto"},{"index":8014,"str":"Immediately the wind filled the sail and Roberto wrapped the boom sheet once around a cleat they had screwed into the middle of the plywood","name":"Roberto"},{"index":8616,"str":"Stefan and Roberto looked at each other round-eyed, and they might even have been nervous if Mr","name":"Roberto"},{"index":8884,"str":"So Roberto kept the sail taut, and Stefan pulled the front skate a little more to the right, pointing them up into the wind a bit more, and they clatterswooshed across the mighty river, which from this vantage appeared like an immense ice lake, like one of the Great Lakes maybe","name":"Roberto"},{"index":9914,"str":"“Amazing!” Roberto said","name":"Roberto"},{"index":10214,"str":"Roberto, white-lipped and already shivering a little, said, “I think we could borrow some gloves and blankets","name":"Roberto"},{"index":10910,"str":"When they had slid until they were pointed toward Manhattan, Roberto pulled the sail taut, and the boat scraped a little sideways downwind, then began hissing and scronching toward the city","name":"Roberto"},{"index":11200,"str":"Roberto hauled with all his might on the sheet and cleated it down before he lost it, and Stefan lay down to stay under the boom, which was now angled over the right side of the boat instead of the left","name":"Roberto"},{"index":11745,"str":"Hexter said to Roberto","name":"Roberto"},{"index":11795,"str":"When Roberto got the sheet loose from the cleat, the sail was freed to flap downwind on the boom, which swung wildly back and forth","name":"Roberto"},{"index":12965,"str":"“This is cool,” Roberto said","name":"Roberto"},{"index":13752,"str":"“What?” Roberto complained","name":"Roberto"},{"index":14586,"str":"Stefan and Roberto stared at him","name":"Roberto"},{"index":15300,"str":"“What’s a smuggled good?” Stefan clarified, glancing at Roberto","name":"Roberto"},{"index":15515,"str":"“And what were you receiving?” Roberto asked","name":"Roberto"},{"index":16825,"str":"“Roberto, shut up!”","name":"Roberto"},{"index":17465,"str":"“Roberto!”","name":"Roberto"},{"index":21058,"str":"“What happened to him?” Roberto asked","name":"Roberto"},{"index":21714,"str":"“But what did he mean about the line being around you?” Roberto asked","name":"Roberto"},{"index":23511,"str":"“Kind of depressing,” Roberto pointed out","name":"Roberto"}]},{"chapter":86,"name":"","length":173,"links":[]},{"chapter":87,"name":"Mutt and Jeff","length":8069,"links":[{"index":0,"str":"Mutt and Jeff sit with Charlotte at their railing, sipping wine from the white coffee cups","name":"Mutt"},{"index":482,"str":"“Arguably we’re saner now than we were before,” Mutt says","name":"Mutt"},{"index":659,"str":"“You did before,” Mutt protests","name":"Mutt"},{"index":744,"str":"Jeff actually smiles at this, pleasing Mutt greatly","name":"Mutt"},{"index":1539,"str":"“Everyone’s full of surprises,” Mutt says","name":"Mutt"},{"index":1794,"str":"“There’s nine billion people on this planet,” Mutt points out, “so actually that’s about one out of every ninety people, if I got my decimal point right","name":"Mutt"},{"index":2016,"str":"“So what did you think?” Mutt inquires of her","name":"Mutt"},{"index":2568,"str":"Charlotte and Mutt don’t get this","name":"Mutt"},{"index":3273,"str":"“But you should have seen him before,” Mutt insists","name":"Mutt"},{"index":4674,"str":"Mutt, thinking to divert Jeff’s no doubt withering critique of their young financier, says, “Have you ever noticed that our building is a kind of actor network that can do things? We got the cloud star, the lawyer, the building expert, the building itself, the police detective, the money man … add the getaway driver and it’s a fucking heist movie!”","name":"Mutt"},{"index":5207,"str":"“Lame-ass jokes,” says Mutt","name":"Mutt"},{"index":5713,"str":"Mutt acknowledges this, but says, “On the other hand, we do have quite a crew here","name":"Mutt"},{"index":6314,"str":"“So it’s mistaking the particular for the general?” Mutt says","name":"Mutt"},{"index":6758,"str":"“Jusvenge,” Mutt tries","name":"Mutt"},{"index":7859,"str":"Mutt feels a smile stretching his face in an almost forgotten way","name":"Mutt"},{"index":0,"str":"Mutt and Jeff sit with Charlotte at their railing, sipping wine from the white coffee cups","name":"Jeff"},{"index":541,"str":"Jeff shakes his head","name":"Jeff"},{"index":744,"str":"Jeff actually smiles at this, pleasing Mutt greatly","name":"Jeff"},{"index":797,"str":"“Delmore Schwartz!” Jeff says","name":"Jeff"},{"index":1655,"str":"“Of course,” says Jeff","name":"Jeff"},{"index":2535,"str":"Jeff says, “We’re all like her","name":"Jeff"},{"index":2603,"str":"Jeff explains: “She wants things to go right","name":"Jeff"},{"index":3425,"str":"Jeff throws his hands in the air, like, What","name":"Jeff"},{"index":3524,"str":"Jeff’s laugh is more like arrrrrgh","name":"Jeff"},{"index":4530,"str":"“Why him?” Jeff asks","name":"Jeff"},{"index":4597,"str":"Jeff shakes his head at her like he’s regarding a true miracle of stupidity","name":"Jeff"},{"index":4674,"str":"Mutt, thinking to divert Jeff’s no doubt withering critique of their young financier, says, “Have you ever noticed that our building is a kind of actor network that can do things? We got the cloud star, the lawyer, the building expert, the building itself, the police detective, the money man … add the getaway driver and it’s a fucking heist movie!”","name":"Jeff"},{"index":5025,"str":"“So who are we?” Jeff says","name":"Jeff"},{"index":5053,"str":"“We are the wise old geezers, Jeffrey","name":"Jeff"},{"index":5093,"str":"“But that’s Gordon Hexter,” Jeff points out","name":"Jeff"},{"index":5570,"str":"“Ease of representation,” Jeff says","name":"Jeff"},{"index":6676,"str":"“Which is one thing,” Jeff reminds her","name":"Jeff"},{"index":6896,"str":"“We demand justice,” Jeff says","name":"Jeff"},{"index":7801,"str":"Jeff cackles along with Charlotte","name":"Jeff"},{"index":7926,"str":"He clicks ceramic cups with Jeff","name":"Jeff"},{"index":4428,"str":"“It would be interesting to hear what Franklin might say about that","name":"Franklin"},{"index":1581,"str":"“Did you hear Amelia Black’s broadcast after her polar bears got nuked?”","name":"Amelia"},{"index":0,"str":"Mutt and Jeff sit with Charlotte at their railing, sipping wine from the white coffee cups","name":"Charlotte"},{"index":692,"str":"Charlotte says, “In dreams begin responsibilities","name":"Charlotte"},{"index":828,"str":"“It’s actually Yeats,” Charlotte explains","name":"Charlotte"},{"index":1229,"str":"Charlotte raises her eyebrows","name":"Charlotte"},{"index":1703,"str":"“It’s got like a hundred million views now,” Charlotte confirms","name":"Charlotte"},{"index":1949,"str":"“That’s everybody,” Charlotte says","name":"Charlotte"},{"index":2063,"str":"Charlotte shrugs","name":"Charlotte"},{"index":2568,"str":"Charlotte and Mutt don’t get this","name":"Charlotte"},{"index":2766,"str":"“We have a plan?” Charlotte suggests","name":"Charlotte"},{"index":3166,"str":"“Euthanasia of the rentier,” Charlotte corrects","name":"Charlotte"},{"index":3382,"str":"“Maybe a little vengeful,” Charlotte says","name":"Charlotte"},{"index":3806,"str":"“The rentier class is not going to go down easily,” says Charlotte","name":"Charlotte"},{"index":4165,"str":"Charlotte considers this","name":"Charlotte"},{"index":5352,"str":"Charlotte shakes her head","name":"Charlotte"},{"index":5798,"str":"Charlotte says, “Everybody does","name":"Charlotte"},{"index":6824,"str":"“Let’s leave it at justice,” Charlotte suggests","name":"Charlotte"},{"index":7607,"str":"“So how does a thing like that start?” Charlotte wonders","name":"Charlotte"},{"index":7801,"str":"Jeff cackles along with Charlotte","name":"Charlotte"}]},{"chapter":88,"name":"","length":163,"links":[]},{"chapter":89,"name":"Vlade","length":11302,"links":[{"index":133,"str":"“We talked to Charlotte about it, and she convinced us to ask Inspector Gen what we should do","name":"Gen"},{"index":0,"str":"Vlade’s wristpad beeped and said, “So how’s it going with our gold?”","name":"Vlade"},{"index":308,"str":"Vlade waited her out","name":"Vlade"},{"index":742,"str":"” Vlade took a deep breath, then gave it a try: “Why don’t you come on over and we’ll talk about it with the gang here","name":"Vlade"},{"index":1497,"str":"Vlade was still finding leaks appearing below the low tide mark on the building, small but worrisome","name":"Vlade"},{"index":1962,"str":"Nervously Vlade welcomed her to the Met and showed her around","name":"Vlade"},{"index":2639,"str":"Idelba seemed impressed, which pleased Vlade","name":"Vlade"},{"index":3351,"str":"“What do you mean?” Vlade was startled by this word","name":"Vlade"},{"index":3448,"str":"Vlade gave Charlotte a call, and as it turned out she was still in the building","name":"Vlade"},{"index":3560,"str":"“This is Idelba,” Vlade said to Charlotte","name":"Vlade"},{"index":3667,"str":"“We were married too,” Idelba said, not knowing that Vlade had told Charlotte about it","name":"Vlade"},{"index":3754,"str":"“Just to help you understand why I would help such a creature as Vlade","name":"Vlade"},{"index":4949,"str":"“Like how I’m helping Vlade work on your place’s security?”","name":"Vlade"},{"index":5162,"str":"You see those hotellos across the farm, we could put up a couple more, right, Vlade?”","name":"Vlade"},{"index":5249,"str":"Vlade tried to imagine what it would be like living near Idelba again, failed, but managed to say “Sure” without much delay","name":"Vlade"},{"index":5598,"str":"Charlotte shrugged, imitating Idelba, Vlade saw","name":"Vlade"},{"index":5808,"str":"Vlade escorted Idelba back down to the boathouse","name":"Vlade"},{"index":6324,"str":"It was painful to Vlade","name":"Vlade"},{"index":6760,"str":"Vlade had had to look it up, but it turned out to be fairly simple","name":"Vlade"},{"index":7921,"str":"Vlade turned off the torch and flipped up the mask","name":"Vlade"},{"index":8441,"str":"Vlade went out and untied the building’s runabout from the boathouse dock, and they got in and hummed out the door into the bacino","name":"Vlade"},{"index":8648,"str":"When they came around from the bacino into the Twenty-fourth canal, Vlade saw that the tug was about half as wide as the canal","name":"Vlade"},{"index":9090,"str":"Vlade pulled up to the tug and Thabo caught the rope Idelba threw up to him and tied them off","name":"Vlade"},{"index":9225,"str":"“Oh my,” Vlade said","name":"Vlade"},{"index":9382,"str":"Idelba called sharply to Thabo and the other man in Berber, and Vlade glimpsed the whites of their eyes before they scrambled belowdecks on the tug","name":"Vlade"},{"index":9729,"str":"“Strong one?” Vlade inquired hopefully","name":"Vlade"},{"index":10129,"str":"“Let’s put your strongbox in a bigger strongbox,” Vlade suggested","name":"Vlade"},{"index":10901,"str":"Vlade tried to think of a snappy reply but failed","name":"Vlade"},{"index":11027,"str":"Truce in the Vlade-Idelba cold war? He would find out later","name":"Vlade"},{"index":133,"str":"“We talked to Charlotte about it, and she convinced us to ask Inspector Gen what we should do","name":"Charlotte"},{"index":1176,"str":"Gathering the treasure consortium was hard, mainly because Charlotte was part of it now, in an advisory role, and she was mostly away, and busy even when she was home","name":"Charlotte"},{"index":3448,"str":"Vlade gave Charlotte a call, and as it turned out she was still in the building","name":"Charlotte"},{"index":3560,"str":"“This is Idelba,” Vlade said to Charlotte","name":"Charlotte"},{"index":3667,"str":"“We were married too,” Idelba said, not knowing that Vlade had told Charlotte about it","name":"Charlotte"},{"index":3828,"str":"“Funny,” Charlotte said, “I was just talking to my ex the other day","name":"Charlotte"},{"index":3923,"str":"Charlotte nodded","name":"Charlotte"},{"index":4030,"str":"Charlotte said, “We’re still trying to figure out how best to maximize its value","name":"Charlotte"},{"index":5009,"str":"Charlotte frowned","name":"Charlotte"},{"index":5598,"str":"Charlotte shrugged, imitating Idelba, Vlade saw","name":"Charlotte"},{"index":7734,"str":"When they deformed and melted together, leaving a scum of the sodium carbonate and dirt on the top, the boys squealed “I’m meltingggg …” which Charlotte had taught them was appropriate","name":"Charlotte"},{"index":6662,"str":"“This is so cool,” Stefan promised Idelba","name":"Stefan"},{"index":6705,"str":"“Even though we shouldn’t be doing it,” Roberto added","name":"Roberto"}]},{"chapter":90,"name":"","length":478,"links":[]},{"chapter":91,"name":"Inspector Gen","length":6643,"links":[{"index":3774,"str":"She settled in and began to apply overlay maps to the snaps of the days when Rosen and Muttchopf had been kidnapped","name":"Mutt"},{"index":5952,"str":"When they left the station she was wondering whether the kidnapping of Rosen and Muttchopf and the bid on the building and attendant sabotages were connected","name":"Mutt"},{"index":0,"str":"After a sudden February thaw Inspector Gen had to take to the skybridges again, having been enjoying her walks on the frozen canals, and she was headed for the one that ran over to One Madison, intending to proceed east from there to the station, when Vlade stopped her at the doors to the skyway","name":"Gen"},{"index":298,"str":"“Hey there Gen, I got something I want to give you","name":"Gen"},{"index":780,"str":"“Sure,” Gen said","name":"Gen"},{"index":2756,"str":"“Maybe so,” Gen said, thinking of the police as a tool","name":"Gen"},{"index":3239,"str":"Late that afternoon Gen went down the hall to the little office carrels inhabited by Claire and Olmstead","name":"Gen"},{"index":3589,"str":"“Don’t let that map fool you,” Gen advised Olmstead","name":"Gen"},{"index":4273,"str":"Gen paused at one point to regard her assistants","name":"Gen"},{"index":4883,"str":"“Pinscher Pinkerton,” Gen said","name":"Gen"},{"index":5414,"str":"“Right,” Gen said","name":"Gen"},{"index":5757,"str":"” Then Gen saw the looks on their faces","name":"Gen"},{"index":5912,"str":"Gen returned to her office to get hers","name":"Gen"},{"index":0,"str":"After a sudden February thaw Inspector Gen had to take to the skybridges again, having been enjoying her walks on the frozen canals, and she was headed for the one that ran over to One Madison, intending to proceed east from there to the station, when Vlade stopped her at the doors to the skyway","name":"Vlade"},{"index":995,"str":"She called Lieutenant Claire and told her to send a boat over to pick up Vlade’s evidence","name":"Vlade"},{"index":1087,"str":"If it was what Vlade thought it was, it might help","name":"Vlade"}]},{"chapter":92,"name":"","length":496,"links":[]},{"chapter":93,"name":"Franklin","length":27982,"links":[{"index":20884,"str":"In your last moment I want you thinking, Damn, I should have listened to that Franklin guy","name":"Franklin"},{"index":4679,"str":"Vlade had put me in touch with his old city teammates’ diving co-op, the Bottom Feeders; they were good to go as divers","name":"Vlade"},{"index":4800,"str":"Vlade’s friend Idelba would serve as dredging subcontractor to them when needed, which, as Hector quickly pointed out, was likely to be often","name":"Vlade"},{"index":9977,"str":"I’ve been working this up with Hector Ramirez and the people in the Met, Charlotte and Vlade and the others","name":"Vlade"},{"index":16051,"str":"I’m going to tell Vlade","name":"Vlade"},{"index":16371,"str":"“Vlade, tie these idiots up, I almost killed them again, they were diving on Twenty-sixth right in the middle of the canal","name":"Vlade"},{"index":16604,"str":"“Sounds a little kinky to me,” Vlade said","name":"Vlade"},{"index":16711,"str":"“Boys,” Vlade said","name":"Vlade"},{"index":16731,"str":"The drowned rats bared their teeth at me and retreated into Vlade’s office","name":"Vlade"},{"index":17006,"str":"When I got there I saw that the boys had dried off and were now sitting in front of Vlade’s screens looking like they were in the principal’s office hoping to get expelled","name":"Vlade"},{"index":17305,"str":"Vlade was working","name":"Vlade"},{"index":18694,"str":"Sullen silence in Vlade’s office","name":"Vlade"},{"index":18727,"str":"Vlade continued to work on his accounts","name":"Vlade"},{"index":18900,"str":"Then, to Charlotte and Vlade: “What the fuck, are these guys wards of the building or not?”","name":"Vlade"},{"index":19627,"str":"He glared at Vlade’s screens","name":"Vlade"},{"index":20560,"str":"“Can you adopt them?” I asked her, but also including Vlade","name":"Vlade"},{"index":20690,"str":"“For why?” Vlade said","name":"Vlade"},{"index":22102,"str":"“Let’s go get that bell,” Vlade suggested heavily to the boys","name":"Vlade"},{"index":22554,"str":"After a while the boys followed Vlade out into the boathouse looking chastened, maybe even thoughtful","name":"Vlade"},{"index":1816,"str":"What does Charlotte Armstrong want to avoid selling a call option on? She doesn’t want there to be an opportunity to buy the Met Life building","name":"Charlotte"},{"index":2046,"str":"What happens if there’s lots of decent housing in the intertidal? It increases a supply, which then decreases the demand on Charlotte’s place","name":"Charlotte"},{"index":9977,"str":"I’ve been working this up with Hector Ramirez and the people in the Met, Charlotte and Vlade and the others","name":"Charlotte"},{"index":16089,"str":"“I’m going to tell Charlotte","name":"Charlotte"},{"index":16514,"str":"“Close enough, so I want to give them over to Charlotte and watch her spank their asses","name":"Charlotte"},{"index":16646,"str":"“And Charlotte is out","name":"Charlotte"},{"index":16868,"str":"I was about to go out again when Charlotte pinged me and I remembered the boys","name":"Charlotte"},{"index":17178,"str":"Charlotte had clearly tired out her eyes by rolling them too much, and was now staring at the ceiling pondering other matters","name":"Charlotte"},{"index":17505,"str":"“City workers,” Charlotte said heavily","name":"Charlotte"},{"index":18297,"str":"“Schliemann at Troy,” Charlotte suggested","name":"Charlotte"},{"index":18824,"str":"Charlotte heaved a big sigh","name":"Charlotte"},{"index":18900,"str":"Then, to Charlotte and Vlade: “What the fuck, are these guys wards of the building or not?”","name":"Charlotte"},{"index":19043,"str":"At this Charlotte pursed her lips","name":"Charlotte"},{"index":20456,"str":"I looked over at Charlotte","name":"Charlotte"},{"index":21617,"str":"“Way,” Charlotte said, transfixing them with her look","name":"Charlotte"},{"index":21735,"str":"“You live here now,” Charlotte reminded them","name":"Charlotte"},{"index":22078,"str":"“Deal,” Charlotte said","name":"Charlotte"},{"index":22446,"str":"Charlotte reached out and put a hand to his arm","name":"Charlotte"},{"index":22658,"str":"I went upstairs with Charlotte","name":"Charlotte"},{"index":25786,"str":"Charlotte leaned over and clinked her glass of water to mine","name":"Charlotte"},{"index":13735,"str":"After that I was humming blindly along when I nearly ran into that Stefan kid, in his same rubber dinghy, looking anxiously over the side as he held an air tube in his hand","name":"Stefan"},{"index":15721,"str":"Stefan hauled up hard on their diving bell’s rope, happily still attached to it this time, and after a while the smaller one appeared from the murky surface of the canal, looking like an otter with a human face","name":"Stefan"},{"index":18014,"str":"“1863 to 1891,” Stefan said","name":"Stefan"},{"index":18436,"str":"“There was a lost manuscript,” Stefan added","name":"Stefan"},{"index":19206,"str":"“We are free citizens of the intertidal,” Stefan asserted","name":"Stefan"},{"index":19297,"str":"“Orphans,” Stefan explained","name":"Stefan"},{"index":19430,"str":"“I grew up with my parents in Russia,” Stefan said","name":"Stefan"},{"index":19657,"str":"Stefan said, “Roberto never had any parents or guardians","name":"Stefan"},{"index":21016,"str":"“What will you be thinking?” Stefan asked","name":"Stefan"},{"index":21822,"str":"“We can still go out and do stuff,” Stefan explained to Roberto","name":"Stefan"},{"index":15315,"str":"Roberto the Reckless","name":"Roberto"},{"index":16145,"str":"Mulishly Roberto pulled himself back on board their rubber boat, and as he shivered bluely I helped them haul up their pathetic diving bellette, then towed them around the corner into the bacino, then into the Met boathouse","name":"Roberto"},{"index":17410,"str":"“It’s not against the law to dive the canals,” Roberto protested","name":"Roberto"},{"index":18411,"str":"“Why not?” Roberto said","name":"Roberto"},{"index":18768,"str":"Feral madness fumed off Roberto like a whiff of skunk","name":"Roberto"},{"index":19590,"str":"“What about you?” I said to Roberto","name":"Roberto"},{"index":19657,"str":"Stefan said, “Roberto never had any parents or guardians","name":"Roberto"},{"index":19779,"str":"Roberto stood up from his chair and said, “I take care of myself","name":"Roberto"},{"index":20805,"str":"“Just don’t say I didn’t warn you when Roberto here goes out there and drowns","name":"Roberto"},{"index":20978,"str":"“Not gonna happen,” Roberto affirmed","name":"Roberto"},{"index":21059,"str":"“Not gonna happen,” Roberto grimly insisted","name":"Roberto"},{"index":21210,"str":"“In lower Manhattan?” Roberto said","name":"Roberto"},{"index":21713,"str":"Even Roberto quailed","name":"Roberto"},{"index":21822,"str":"“We can still go out and do stuff,” Stefan explained to Roberto","name":"Roberto"},{"index":21916,"str":"Roberto looked at the floor","name":"Roberto"}]},{"chapter":94,"name":"","length":164,"links":[]},{"chapter":95,"name":"Charlotte","length":15548,"links":[{"index":10916,"str":"Gentrification, enclosure, whatever you want to call it","name":"Gen"},{"index":0,"str":"Charlotte found herself actually pleased to be giving her ex Larry a call to ask for another coffee date","name":"Charlotte"},{"index":1267,"str":"On the appointed day, Charlotte got in one of the cable cars running up thick steel lines from the East Village to the Brooklyn Bridge’s western tower","name":"Charlotte"},{"index":3246,"str":"It was one of the great views of the city, and Charlotte suspected Larry had chosen it with her in mind, thinking it would please her, which it did","name":"Charlotte"},{"index":3790,"str":"Icy-cold vodka was a drink Charlotte despised, but it helped wash down the even weirder oyster flavor","name":"Charlotte"},{"index":4147,"str":"Over their meal, which consisted of Cobb salads for both of them, way better than anything the Met’s kitchen could assemble, Charlotte tacked her way to the point of this meeting","name":"Charlotte"},{"index":6958,"str":"“So look,” Charlotte said, and leaned toward him","name":"Charlotte"},{"index":8063,"str":"Charlotte saw that and almost laughed, but instead she kept her focus and pounced:","name":"Charlotte"},{"index":9375,"str":"Larry shook his head, trying for one of his old expressions, this one meant to express faked admiration for Charlotte, an expression she remembered very well","name":"Charlotte"},{"index":9670,"str":"“Yeah yeah, Red Charlotte","name":"Charlotte"},{"index":9698,"str":"“Charlotte Corday, isn’t that right?”","name":"Charlotte"},{"index":10387,"str":"Charlotte had to admit it: her ex was smart","name":"Charlotte"},{"index":12197,"str":"Charlotte Corday is already bad enough","name":"Charlotte"},{"index":13898,"str":"Spring was springing, right before their eyes; looking around at the flushed faces Charlotte saw that it was an erotic and even a sexual high, a March madness indeed","name":"Charlotte"}]},{"chapter":96,"name":"","length":689,"links":[]},{"chapter":97,"name":"Citizen","length":6005,"links":[]},{"chapter":98,"name":"","length":461,"links":[]},{"chapter":99,"name":"Stefan and Roberto","length":12334,"links":[{"index":3002,"str":"At first they were disappointed it wasn’t under water, but as they had given over their diving bell to Vlade, they became reconciled and decided it was a good thing","name":"Vlade"},{"index":6263,"str":"And there are jobs down there, like what Vlade did","name":"Vlade"},{"index":1430,"str":"Stefan and Roberto didn’t care","name":"Stefan"},{"index":4142,"str":"“It’s like something you would do,” Stefan noted","name":"Stefan"},{"index":4956,"str":"“It’s big,” Stefan said","name":"Stefan"},{"index":7045,"str":"“Thunderhead?” Stefan said","name":"Stefan"},{"index":8845,"str":"“We should have waited it out!” Stefan shouted as one particularly big white wall tilted them almost vertically before it passed under them, and the bow then flopped down so hard they had to hold on to avoid being tossed forward","name":"Stefan"},{"index":9183,"str":"Stefan didn’t reply, but it was true","name":"Stefan"},{"index":10144,"str":"Seeing his look, Stefan grew afraid","name":"Stefan"},{"index":10672,"str":"As the bow dropped forward under the impact of their bodies, Roberto twisted the tiller toward Stefan, and as the boat slid down the back side of the wave he gunned the motor to its max","name":"Stefan"},{"index":12268,"str":"“Looks like it,” Stefan agreed","name":"Stefan"},{"index":1430,"str":"Stefan and Roberto didn’t care","name":"Roberto"},{"index":4077,"str":"“Oh man,” Roberto said","name":"Roberto"},{"index":4981,"str":"“We should take it back to his neighborhood,” Roberto insisted","name":"Roberto"},{"index":5958,"str":"Roberto sighed","name":"Roberto"},{"index":6659,"str":"” Roberto rubbed his hand over Melville’s gravestone, thinking it over","name":"Roberto"},{"index":10106,"str":"Roberto was staring ahead, round-eyed","name":"Roberto"},{"index":10402,"str":"“Next one,” Roberto said","name":"Roberto"},{"index":10672,"str":"As the bow dropped forward under the impact of their bodies, Roberto twisted the tiller toward Stefan, and as the boat slid down the back side of the wave he gunned the motor to its max","name":"Roberto"},{"index":11007,"str":"The broken wall of water hit when they were about three quarters turned to it, and Roberto pulled on the tiller so that as the boat skidded forward it straightened in orientation to the wave, the stern rising slower than the bow had, they were in the broken foam and it seemed they would be swamped, but aside from a splashing they were spared, as the boat was buoyant and the wave orderly","name":"Roberto"},{"index":12150,"str":"“We’re going to make it,” Roberto declared","name":"Roberto"}]},{"chapter":100,"name":"","length":91,"links":[]},{"chapter":101,"name":"Vlade","length":56493,"links":[{"index":52569,"str":"Inspector Gen had been out working since the storm began, but she had come back home the night before on a police cruiser, to change clothes and catch a couple of hours of sleep","name":"Gen"},{"index":55715,"str":"“What?” Franklin exclaimed","name":"Franklin"},{"index":56337,"str":"“I’m busy too!” Franklin exclaimed","name":"Franklin"},{"index":0,"str":"As part of his job Vlade kept the NOAA weather page for New York up on one of his screens, in a box next to the tide screen","name":"Vlade"},{"index":841,"str":"Vlade had a page in his files for stormproofing the building, and he called it up on his main screen and alerted his team: all hands on deck! The to-do list was long and they would have to hurry","name":"Vlade"},{"index":1036,"str":"Not a drill, Vlade told his team","name":"Vlade"},{"index":2468,"str":"“Hey Vlade, what’s up?”","name":"Vlade"},{"index":3326,"str":"Vlade shook his head","name":"Vlade"},{"index":3611,"str":"The Met was now around 230 years old, although to Vlade this meant little","name":"Vlade"},{"index":3991,"str":"Vlade had no fear that anything could bring the tower down, it was foursquare and massively reinforced","name":"Vlade"},{"index":4183,"str":"Indeed if either of those stupid toothpicks fell north they could wreck the Met too, a thought that gave Vlade the creeps","name":"Vlade"},{"index":4306,"str":"Hopefully if they fell it would be in some other direction, although if they fell west they would crush the Flatiron, a building everyone around the bacino loved, though Vlade was glad he wasn’t its super; all those nonsquared walls were a pain, as Ettore was always saying, especially the narrow point at the north, where dogs had to wag their tails up and down, as Ettore had it","name":"Vlade"},{"index":5396,"str":"“People!” Vlade said to them peremptorily","name":"Vlade"},{"index":5796,"str":"Vlade frowned","name":"Vlade"},{"index":5911,"str":"This was no easy task, and few of them had ever done it, so Vlade had to educate people to the system","name":"Vlade"},{"index":7079,"str":"So Vlade joined a conference call with the local gridmaster, who was coordinating plans for various flex contingencies across the board, from total retention to complete loss, with the latter possibility taking up most of the talk","name":"Vlade"},{"index":8577,"str":"Vlade didn’t say that out loud, but other supers did; this was New York, after all","name":"Vlade"},{"index":9014,"str":"A call came in: “Vlade, it’s Amelia","name":"Vlade"},{"index":10228,"str":"“Vlade!”","name":"Vlade"},{"index":10237,"str":"“Don’t Vlade me! Deal!”","name":"Vlade"},{"index":11438,"str":"“Let’s get everything we can off the farm floor,” Vlade said to his team","name":"Vlade"},{"index":12427,"str":"“Fuck,” Vlade kept saying","name":"Vlade"},{"index":12698,"str":"Vlade felt his stomach drop","name":"Vlade"},{"index":13005,"str":"Vlade called their wristpad and got no answer","name":"Vlade"},{"index":13799,"str":"Vlade went back up to the top of the building and made sure everything under the cupola was secured, feeling as grim as Quasimodo","name":"Vlade"},{"index":14708,"str":"Vlade took the elevator back up to the cupola, then climbed the spiral stairs to the blimp room, where narrow windows gave a view from as high as the building afforded","name":"Vlade"},{"index":15843,"str":"“Wow!” Vlade shouted","name":"Vlade"},{"index":18997,"str":"There was no way to fully close the boathouse off from the bacino itself, which was something Vlade promised himself to change in the future","name":"Vlade"},{"index":21419,"str":"The roar of the wind and rain filled everything, and as people had to shout to be heard, they shouted all the more to surmount their own din, in the usual party style, until Vlade felt like it was time to get back to the relative quiet of the control room","name":"Vlade"},{"index":21942,"str":"Vlade got next to the window and fearfully looked up; it was just possible to discern the water level, up there near the ceiling, crowded with the hulls of boats from the lowest two levels of his sling rafters, all banging around up there in the surface slop together","name":"Vlade"},{"index":26668,"str":"Vlade recrossed the skybridge, pausing out in the middle to look around again","name":"Vlade"},{"index":27154,"str":"Feeling spooked, and awed, Vlade got back into the Met","name":"Vlade"},{"index":27649,"str":"Vlade passed some time tapping out various scenarios on his spreadsheets, using Gantt programs to see how they might do","name":"Vlade"},{"index":28265,"str":"The other buildings in the neighborhood were mostly okay, but one of the bishop skybridges between the Decker building and the New School had come down over Fifth and Fourteenth, and both buildings were now coping with open holes in their sides, just as Vlade had expected","name":"Vlade"},{"index":29430,"str":"“Incredible,” Vlade said when he saw that","name":"Vlade"},{"index":30713,"str":"Vlade wondered how Brooklyn was doing but didn’t bother to look into it","name":"Vlade"},{"index":31473,"str":"Later when they were alone Vlade said, “I’d feel a lot better if I knew where they were","name":"Vlade"},{"index":31994,"str":"Vlade sighed","name":"Vlade"},{"index":32122,"str":"Vlade spent some time looking for ways to cut more power without making people uncomfortable","name":"Vlade"},{"index":33739,"str":"The center of the hurricane passed in the night, and there was the classic lull that occurred at the eye of the storm, audible even from Vlade’s bed, in the negative sense that the background roar went away for a while","name":"Vlade"},{"index":33958,"str":"Barometer reading crazy low, it bottomed out on Vlade’s barometer at 25","name":"Vlade"},{"index":34342,"str":"So Vlade and Idelba got up and climbed the stairs to the tower again to have a look","name":"Vlade"},{"index":35046,"str":"Vlade, focused on the problems of keeping the Met secure, was shocked at the notion that anything could be done","name":"Vlade"},{"index":35599,"str":"Vlade sighed","name":"Vlade"},{"index":37836,"str":"They had two things going for them in the city, counterintuitive though both seemed to Vlade: the canals were so narrow and shallow that the water in them could only become a chaos of blown spray and froth, without high waves; in effect the waves were being blown off or smashed flat","name":"Vlade"},{"index":39359,"str":"Idelba said to Vlade, “Can you go out and help get people on board?”","name":"Vlade"},{"index":39428,"str":"Vlade nodded, took a deep breath, and left the bridge, using the door on its north side","name":"Vlade"},{"index":40515,"str":"They waved to Vlade from broken windows or even lying flat on rooftops, and as the tug motored down Second, Vlade indicated left or right, and Idelba and her guys got the tug over next to the buildings, and people jumped onto the tug, sometimes dropping ten feet or more, which of course injured many of them","name":"Vlade"},{"index":41254,"str":"The tug had an open deck, but Vlade got people tucked under the high taffrails and sent the worst into the cabins under the bridge, although he didn’t like opening those doors","name":"Vlade"},{"index":41762,"str":"“Bellevue hospital is at Twenty-sixth and First,” Vlade said","name":"Vlade"},{"index":42090,"str":"Vlade leaped back out into the onslaught","name":"Vlade"},{"index":43677,"str":"Vlade had called ahead, and many of the Met’s residents were there to help the remaining passengers into the building","name":"Vlade"},{"index":43866,"str":"“We’ll run out of fuel in about five runs,” she shouted to Vlade when he came into the bridge","name":"Vlade"},{"index":44060,"str":"Vlade wondered if any fuel depots would still be operating","name":"Vlade"},{"index":44865,"str":"“Damn, this is just like that overtopping of Bjarke’s Wall that Hexter told us about,” Vlade said to no one, looking up and down the whitewater canals","name":"Vlade"},{"index":45474,"str":"Vlade could only nod and go back outside into the storm to help people get over the side of the tug, and into the cabins if they were hurt","name":"Vlade"},{"index":45796,"str":"By standing on an awning frame they could just make it high enough for Vlade to help boost them up and over the side","name":"Vlade"},{"index":46138,"str":"Just as Vlade was leaning down to grab the hands of the reaching men, a big wave caught the tug from the left, possibly a surge from a fallen building, anyway massive; it cast the tug right into the building at the corner, crushing the two men between tug and wall with a palpable thump","name":"Vlade"},{"index":46426,"str":"The tug held there against the wall, and Vlade, who had jerked up just in time to get clear of the collision himself, looked up at Idelba and screamed at her to turn left, waving his arms desperately","name":"Vlade"},{"index":46944,"str":"Vlade looked down; the two men were gone","name":"Vlade"},{"index":48219,"str":"But then, as they were headed for the Met, the wind subsided to a mere gale, maybe thirty miles an hour, Vlade guessed; so Idelba kept them going, the tug’s super-powerful night lights glazing the immediate vicinity like a welder’s torch","name":"Vlade"},{"index":48868,"str":"By the time they called it quits they had put a couple thousand people into the hospitals, Vlade reckoned, and another thousand into the Met","name":"Vlade"},{"index":49624,"str":"“Damn, I wish I knew where those boys were,” Vlade said as he was falling asleep on his bed, Idelba out on the couch in his office","name":"Vlade"},{"index":49923,"str":"And then Vlade was out","name":"Vlade"},{"index":50656,"str":"Vlade could now get back into his boathouse, and so he unsealed the door to it and began to sort out the confusion created by having all the boats floated up into each other, and in some cases crushed a bit against the ceiling","name":"Vlade"},{"index":51430,"str":"The health challenges were going to be severe, Vlade saw; it was already warm again, and cholera was all too likely","name":"Vlade"},{"index":52431,"str":"“Also, we’re out of food,” Vlade told them, which was close enough to true to allow him to say it","name":"Vlade"},{"index":53103,"str":"Vlade had never seen the inspector look as tired as she did now, and this was just the start of it","name":"Vlade"},{"index":53439,"str":"“Held up fine,” Vlade said","name":"Vlade"},{"index":53596,"str":"“Jesus!” Vlade said","name":"Vlade"},{"index":55409,"str":"Vlade stumped back down to the common room and shared the news","name":"Vlade"},{"index":56372,"str":"But then he saw Vlade’s look and said, “All right, all right, I’ll go have a look","name":"Vlade"},{"index":1316,"str":"He was leaving his office to get started on the stormproofing when he recalled that Amelia Black was out there in the air somewhere nearby, and Idelba was on her barge off Coney Island","name":"Amelia"},{"index":2455,"str":"Now Amelia","name":"Amelia"},{"index":2492,"str":"“Amelia, where are you?”","name":"Amelia"},{"index":2730,"str":"“Amelia, how far can you see south?”","name":"Amelia"},{"index":3284,"str":"“Amelia!”","name":"Amelia"},{"index":9014,"str":"A call came in: “Vlade, it’s Amelia","name":"Amelia"},{"index":9576,"str":"“Damn it Amelia, this is going to be big, do you understand?”","name":"Amelia"},{"index":13929,"str":"The boys were missing, and Amelia was flying the storm in a blimp","name":"Amelia"},{"index":12726,"str":"“Stefan and Roberto?”","name":"Stefan"},{"index":31352,"str":"And of course that made them both think of Stefan and Roberto","name":"Stefan"},{"index":31902,"str":"“Or that Stefan will stop him from doing anything too stupid","name":"Stefan"},{"index":55295,"str":"If Roberto and Stefan came back in safe and sound, all would be well","name":"Stefan"},{"index":12726,"str":"“Stefan and Roberto?”","name":"Roberto"},{"index":31352,"str":"And of course that made them both think of Stefan and Roberto","name":"Roberto"},{"index":31762,"str":"“Roberto is not too good at risk assessment,” he said","name":"Roberto"},{"index":55295,"str":"If Roberto and Stefan came back in safe and sound, all would be well","name":"Roberto"}]},{"chapter":102,"name":"","length":53,"links":[]},{"chapter":103,"name":"Inspector Gen","length":16327,"links":[{"index":15623,"str":"“Maybe you’ve found the infiltration that made it possible for them to disable the cameras when they snatched Mutt and Jeff","name":"Mutt"},{"index":15623,"str":"“Maybe you’ve found the infiltration that made it possible for them to disable the cameras when they snatched Mutt and Jeff","name":"Jeff"},{"index":0,"str":"Gen got the call like every other police officer in the tri-state area: emergency, all hands on deck","name":"Gen"},{"index":980,"str":"The need for it was so great that the cruiser carrying Gen had to wait its turn, and then they all disembarked in a hurry","name":"Gen"},{"index":1104,"str":"Walking up into Central Park, Gen was amazed by what she saw","name":"Gen"},{"index":2300,"str":"Gen listened to the police officers already on hand and took their point: the first order of business was to get the crowds who were doing cleanup work to consider their own safety and desist","name":"Gen"},{"index":2827,"str":"“We’ve had enough injuries already,” Gen said over and over","name":"Gen"},{"index":4730,"str":"Gen got on the wrist with headquarters and made the same reports and requests as everyone else, judging by the responses she got","name":"Gen"},{"index":4889,"str":"“Are the feds coming?” Gen asked","name":"Gen"},{"index":4944,"str":"Gen went over to the Wollman ice rink, where it seemed like they should be able to clear an area big enough for even the largest helicopters to land","name":"Gen"},{"index":6308,"str":"As she wandered the park doing what she could, Gen started making lists in her head, redundant lists, as obviously the various emergency services already had them, but she couldn’t help herself","name":"Gen"},{"index":9283,"str":"It was against the law to have fires in the park, but Gen waved at them","name":"Gen"},{"index":9798,"str":"Gen kept doing the work of a beat cop, and there were definitely more problems to conciliate, more petty crimes reported","name":"Gen"},{"index":12872,"str":"Gen shook her head","name":"Gen"},{"index":13065,"str":"The man hesitated and Gen gestured to her team, and four more officers leaped over the sides of the boat, holstered guns prominently displayed","name":"Gen"},{"index":13323,"str":"Gen, the only woman on the boat, also the person in charge, kept a straight face, kept it professional and polite","name":"Gen"},{"index":13942,"str":"“Investment,” Gen repeated","name":"Gen"},{"index":14198,"str":"“We’re private investment security,” the man explained when Gen looked up at him","name":"Gen"},{"index":14335,"str":"“Maybe so,” Gen said","name":"Gen"},{"index":15606,"str":"“Okay!” Gen said","name":"Gen"},{"index":11013,"str":"Vlade awarded her a shower","name":"Vlade"},{"index":15235,"str":"“So, you know who else used to work for Escher? Three of the people now working at the Met tower for Vlade Marovich","name":"Vlade"}]},{"chapter":104,"name":"","length":85,"links":[]},{"chapter":105,"name":"Franklin","length":10755,"links":[{"index":174,"str":"But it wasn’t just Vlade asking me in his heavy Slavic-mafia way, gravid or even morbid with the responsibility for all the creatures in his ark, including yea the littlest and most stupid among them","name":"Vlade"},{"index":1219,"str":"Vlade pulled my bug down from the rafters and glowered me out the door","name":"Vlade"},{"index":374,"str":"It was also Charlotte","name":"Charlotte"},{"index":1291,"str":"I was pleased to get out, of course, and didn’t want Charlotte to think I was unwilling to help","name":"Charlotte"},{"index":4747,"str":"A call came to my wrist from Charlotte","name":"Charlotte"},{"index":6301,"str":"Stefan swallowed and said, “Thanks for looking for us, Mr","name":"Stefan"},{"index":6686,"str":"Stefan swallowed again and drank deeply from a glass of water","name":"Stefan"},{"index":7192,"str":"“Yeah and a lot of them broke outward!” Stefan said","name":"Stefan"},{"index":7525,"str":"“That happened,” Stefan confirmed","name":"Stefan"},{"index":8326,"str":"“There were a lot of rats and insects too,” Stefan added after swallowing","name":"Stefan"},{"index":9018,"str":"“Actually the muskrats were wondering if they could eat us,” Stefan said","name":"Stefan"},{"index":9790,"str":"Stefan nodded","name":"Stefan"},{"index":6229,"str":"“Glad to see you too,” Roberto mumbled through a mouthful of something","name":"Roberto"},{"index":7127,"str":"“Windows broke,” Roberto added between chews","name":"Roberto"},{"index":8049,"str":"By way of further answer Roberto pointed at his cheek","name":"Roberto"},{"index":8462,"str":"“In that there were lots of things creeping,” Roberto clarified","name":"Roberto"},{"index":8692,"str":"“Not otters,” Roberto said","name":"Roberto"},{"index":9221,"str":"“It was pretty funny,” Roberto confirmed","name":"Roberto"},{"index":10721,"str":"“Any more cake?” Roberto inquired","name":"Roberto"}]},{"chapter":106,"name":"","length":130,"links":[]},{"chapter":107,"name":"Citizen","length":6250,"links":[]},{"chapter":108,"name":"","length":334,"links":[]},{"chapter":109,"name":"Charlotte","length":17409,"links":[{"index":12419,"str":"It’s the first verse of Genesis","name":"Gen"},{"index":160,"str":"Franklin had gone off to hunt for Stefan and Roberto, looking so worried that she had been tempted to accompany him, but it wouldn’t have helped, and she wanted to do something helpful","name":"Franklin"},{"index":9271,"str":"Franklin Garr walked into the room, headed for the food line","name":"Franklin"},{"index":11257,"str":"Vlade melted it and I sold it, in increments, in various dark pools","name":"Vlade"},{"index":0,"str":"Charlotte went back to work, not knowing what else to do, and figuring that the Householders’ Union office was going to be inundated with new internal refugees","name":"Charlotte"},{"index":703,"str":"Charlotte wasn’t sure that wasn’t true; they were part of a very large crowd now","name":"Charlotte"},{"index":4394,"str":"Get real, Charlotte","name":"Charlotte"},{"index":5002,"str":"I’m busy Charlotte, I have to go","name":"Charlotte"},{"index":5087,"str":"“Fuck you!” Charlotte shouted at her wrist","name":"Charlotte"},{"index":5320,"str":"Charlotte gritted her teeth","name":"Charlotte"},{"index":6214,"str":"But after every crisis of the last century, Charlotte thought, or maybe forever, capital had tightened the noose around the neck of labor","name":"Charlotte"},{"index":6816,"str":"Finally Charlotte gave up thinking and began wandering the park again, stopping to talk to people huddled around the various smoky fires, which existed more for cooking than warmth, or just to be doing something","name":"Charlotte"},{"index":7703,"str":"“Hey Ramona, Charlotte here","name":"Charlotte"},{"index":9333,"str":"“Hey Frankie!” Charlotte said","name":"Charlotte"},{"index":10449,"str":"“So listen,” Charlotte said","name":"Charlotte"},{"index":15053,"str":"Charlotte looked at him, trying to figure out how serious he was","name":"Charlotte"},{"index":16201,"str":"“I’m not sure that’s its main illegality,” Charlotte said","name":"Charlotte"},{"index":16590,"str":"“Hmm,” Charlotte said","name":"Charlotte"},{"index":17205,"str":"“Fuck,” Charlotte complained","name":"Charlotte"},{"index":160,"str":"Franklin had gone off to hunt for Stefan and Roberto, looking so worried that she had been tempted to accompany him, but it wouldn’t have helped, and she wanted to do something helpful","name":"Stefan"},{"index":160,"str":"Franklin had gone off to hunt for Stefan and Roberto, looking so worried that she had been tempted to accompany him, but it wouldn’t have helped, and she wanted to do something helpful","name":"Roberto"}]},{"chapter":110,"name":"","length":428,"links":[]},{"chapter":111,"name":"Inspector Gen","length":22865,"links":[{"index":0,"str":"Gen worked overtime day after day","name":"Gen"},{"index":1211,"str":"Some of them said it looked like gang-on-gang violence, but when Gen got there, coming up on a packed police cruiser, she couldn’t see anything resembling sides; it was just a scramble, knots of people roaming the park, roaring, setting fires with brands from the big bonfire, throwing burning brands, and fighting other groups","name":"Gen"},{"index":1924,"str":"“We know,” said Chief Quinn Taller, an acquaintance of Gen’s","name":"Gen"},{"index":2026,"str":"“They’re headed uptown?” Gen said","name":"Gen"},{"index":2262,"str":"Gen took a deep breath","name":"Gen"},{"index":2595,"str":"Gen got off","name":"Gen"},{"index":4082,"str":"Gen was relieved to be with other cops","name":"Gen"},{"index":4887,"str":"Nick Park and were hurrying up the shore path at the high tide mark, still a shambles of wrack from the storm surge, when a branch hurtled out of the dark and struck the cop right next to Gen on the head","name":"Gen"},{"index":5511,"str":"First aid in the dark, Gen working the downed cop, the rest bulling around ordering dispersal, angry but lacking any way to take it out appropriately","name":"Gen"},{"index":7240,"str":"Electric power seemed to have gone out up here, and Gen wondered if that had started the riot","name":"Gen"},{"index":7558,"str":"The group of cops Gen was part of tried to figure out what to do next, but it was too loud to talk, too loud to think","name":"Gen"},{"index":8346,"str":"Then Gen saw it clearly, and maybe everyone did: it was all about the towers","name":"Gen"},{"index":8609,"str":"Gen’s ad hoc platoon stumbled with the crowd itself into the great plaza south of Amsterdam and 133rd, where the first big cluster of towers shot up to their impossible height, scoring a moony gray sky, looking like space elevators","name":"Gen"},{"index":9907,"str":"Gen had just gotten a vest and helmet on when she heard shots ring out","name":"Gen"},{"index":10736,"str":"Gen spoke into her wrist fiercely: “We need more support! There’s private security here who have opened fire on the crowd!”","name":"Gen"},{"index":11099,"str":"Gen walked over and joined a group of about ten police officers in vests who were headed up the broad steps toward the security forces on the highest terrace","name":"Gen"},{"index":11715,"str":"Gen pulled her pistol from its holster, feeling her skin go hot all over as she did so","name":"Gen"},{"index":12838,"str":"Gen figured she might be the senior officer there, and in any case no one else was doing it, so she walked forward from out of the other cops, pistol extended down to the side","name":"Gen"},{"index":13256,"str":"He looked familiar to her, and he seemed to recognize Gen as well","name":"Gen"},{"index":13324,"str":"“What the fuck were you doing shooting off those guns?” Gen said to him","name":"Gen"},{"index":13469,"str":"Gen waited a beat, then slowly stepped toward the man","name":"Gen"},{"index":13966,"str":"“Stand down and get inside your buildings,” Gen said to the man, staring hard at him","name":"Gen"},{"index":14166,"str":"“You were the first people to fire guns tonight,” Gen told the man","name":"Gen"},{"index":15041,"str":"Gen said, “Come on, inside","name":"Gen"},{"index":16283,"str":"Once inside, Gen stuck to the man and asked him to sit down in the lobby with her","name":"Gen"},{"index":16888,"str":"Gen suddenly recalled where she had seen him before","name":"Gen"},{"index":18249,"str":"Gen nodded","name":"Gen"},{"index":18603,"str":"Now Gen shook her head","name":"Gen"},{"index":19020,"str":"Gen snorted","name":"Gen"},{"index":19135,"str":"” Gen stood back up, looked down at the man","name":"Gen"},{"index":19534,"str":"Gen sighed","name":"Gen"},{"index":20824,"str":"Five minutes ago this guy had been thinking Gen was just a local cop","name":"Gen"},{"index":20954,"str":"Maybe he was recalling better the encounter with Gen on the boat downtown","name":"Gen"},{"index":21248,"str":"Gen gestured at the couches, sat back down","name":"Gen"},{"index":21386,"str":"Now it was Gen’s turn to be surprised","name":"Gen"},{"index":21603,"str":"Might need Gen as a pseudo-ally, somehow, somewhere","name":"Gen"},{"index":22239,"str":"Gen was beginning to think that this night might not have been a complete fucking disaster after all, when the sound of gunfire erupted outside","name":"Gen"},{"index":22432,"str":"Gen surveyed the lobby, the little militia she was in here with","name":"Gen"},{"index":22736,"str":"Gen shrugged","name":"Gen"}]},{"chapter":112,"name":"","length":934,"links":[]},{"chapter":113,"name":"Amelia","length":19052,"links":[{"index":12829,"str":"Charlotte and Franklin and Vlade all said hi at once, sounding relieved she had answered","name":"Franklin"},{"index":13367,"str":"“We’ll help you,” Franklin said loudly over the clatter of their voices","name":"Franklin"},{"index":13533,"str":"“Why not?” Franklin said","name":"Franklin"},{"index":15940,"str":"Amelia took a deep breath, listened to the voices chattering desperately in her ear: Charlotte and Franklin in rapid counterpoint, having a little real-time editing war over what she should say","name":"Franklin"},{"index":398,"str":"Once Vlade had alerted her to the situation, she and Frans had done the right things, all with her broadcasting the adventure to her audience in the cloud, which grew by the minute as people heard what she had gotten herself into this time","name":"Vlade"},{"index":809,"str":"But from the moment Vlade had alerted her to the danger, she had flown the Assisted Migration north as fast as it would go, and although this top speed was only fifty miles an hour in still air, with a growing tailwind pushing her it had been enough to get her to the little town of Hudson, New York, which she called Hudson on the Hudson, where she was allowed to tie off on one of the blimp masts at the Marina Abramovic Institute, named after one of her heroes and role models","name":"Vlade"},{"index":8237,"str":"Amelia called home to tell Vlade she was going to make a circuit or two before coming in","name":"Vlade"},{"index":8561,"str":"“People charged them last night,” Vlade said","name":"Vlade"},{"index":9441,"str":"“Okay,” Vlade said","name":"Vlade"},{"index":9574,"str":"“Fish,” Vlade said","name":"Vlade"},{"index":12829,"str":"Charlotte and Franklin and Vlade all said hi at once, sounding relieved she had answered","name":"Vlade"},{"index":0,"str":"The next day, July 8, 2142, Amelia Black floated down the Hudson River Valley toward home","name":"Amelia"},{"index":639,"str":"Amelia Errorheart has done it again, Amelia Errhard is in big trouble, Amelia Blank is blanking again, Amelia Airhead might not be able to read a map, ha ha, et cetera","name":"Amelia"},{"index":1289,"str":"Once the airship was tied to that mast, its intense flailing became a natural piece of performance art, and at first Amelia had resolved to stay in the gondola through the hurricane—tie herself into a chair and get tossed around like a bull rider, like Marina herself doing one of her variously dangerous and awesome performances; she would be riding the storm! as she put it to her fans","name":"Amelia"},{"index":1678,"str":"But even with the spirit of their founder hovering over the institute and encouraging Amelia to go for it, the actual curators of the place had insisted that given the forecast, in this instance discretion was the better part of value, as they liked having Amelia there but didn’t want her getting thrashed to death witnessed by millions in the cloud","name":"Amelia"},{"index":2297,"str":"“I am fully mentally capacitated,” Amelia objected","name":"Amelia"},{"index":2509,"str":"So, Amelia had with some difficulty gotten out of the gondola without getting crushed under it, and after that watched Frans ride out the storm, narrating the spectacle from inside the institute","name":"Amelia"},{"index":3629,"str":"So Amelia would have been safer in the gondola than in any building whatsoever, another testament to the Assisted Migration, also to the principle of flexibility, of soft power and adaptation, so superior to rigidity and hard power, as she pointed out while narrating the admittedly still very dramatic images of the Assisted Migration shimmying like a shape-shifter under the storm’s wicked slaps","name":"Amelia"},{"index":4706,"str":"Amelia and her hosts weren’t the only people in trouble, nor among those in the worst trouble","name":"Amelia"},{"index":5259,"str":"So Frans was deflated and thrashed on the ground like a big carpet, and there were repairs to be made before Amelia could return to the air, but eventually it got done by the ground crew of a nearby airfield, happy to get the famous cloud star back in the air (and themselves briefly in the cloud with her)","name":"Amelia"},{"index":6283,"str":"“Wow,” Amelia said to her viewers","name":"Amelia"},{"index":7368,"str":"People like ants everywhere, the lost ones of the city huddling there, mostly out of an instinct to huddle, it seemed to Amelia","name":"Amelia"},{"index":8237,"str":"Amelia called home to tell Vlade she was going to make a circuit or two before coming in","name":"Amelia"},{"index":8920,"str":"“Hi Amelia!” came the voice of Roberto","name":"Amelia"},{"index":9400,"str":"How did our building do?” Amelia asked","name":"Amelia"},{"index":9758,"str":"“I don’t like that,” Amelia said","name":"Amelia"},{"index":13487,"str":"“Yay,” Amelia said","name":"Amelia"},{"index":13604,"str":"So listen, Amelia, just say it your way, and if you seem to be having trouble, pause and listen to the voices in your ear, and we’ll feed you lines","name":"Amelia"},{"index":13755,"str":"“Good,” Amelia said","name":"Amelia"},{"index":15940,"str":"Amelia took a deep breath, listened to the voices chattering desperately in her ear: Charlotte and Franklin in rapid counterpoint, having a little real-time editing war over what she should say","name":"Amelia"},{"index":16134,"str":"Amelia just repeated whatever sounded good to her in what she managed to catch of their discourse","name":"Amelia"},{"index":16856,"str":"Still just Amelia Black","name":"Amelia"},{"index":17149,"str":"It’s not just silly Amelia making another bonehead move—I mean, wait here just a second …","name":"Amelia"},{"index":18938,"str":"That is so often true! So that’s it for this episode of Assisted Migration with Amelia Black","name":"Amelia"},{"index":8521,"str":"“We hear they’re okay,” Charlotte said","name":"Charlotte"},{"index":8804,"str":"Charlotte said, “That’s what I was thinking","name":"Charlotte"},{"index":9703,"str":"“Not the people in the superscrapers,” Charlotte said","name":"Charlotte"},{"index":12829,"str":"Charlotte and Franklin and Vlade all said hi at once, sounding relieved she had answered","name":"Charlotte"},{"index":13507,"str":"“Really?” Charlotte said","name":"Charlotte"},{"index":15940,"str":"Amelia took a deep breath, listened to the voices chattering desperately in her ear: Charlotte and Franklin in rapid counterpoint, having a little real-time editing war over what she should say","name":"Charlotte"},{"index":8960,"str":"“Roberto! Stefan, are you there too?”","name":"Stefan"},{"index":9148,"str":"“We talked them out of it,” Stefan said","name":"Stefan"},{"index":8920,"str":"“Hi Amelia!” came the voice of Roberto","name":"Roberto"},{"index":8960,"str":"“Roberto! Stefan, are you there too?”","name":"Roberto"},{"index":9076,"str":"“We almost got eaten by muskrats,” Roberto said","name":"Roberto"}]},{"chapter":114,"name":"","length":1237,"links":[]},{"chapter":115,"name":"Citizen","length":5238,"links":[{"index":1006,"str":"General strikes and people massing in urban centers are usually understood to be the classic forms of civil resistance, but all the other methods listed above fit the definition, and have been effective in the past","name":"Gen"}]},{"chapter":116,"name":"","length":81,"links":[]},{"chapter":117,"name":"Mutt and Jeff","length":8163,"links":[{"index":563,"str":"Mutt hands over the bag and contemplates his partner hefting hammer and nails","name":"Mutt"},{"index":1108,"str":"“Or you could type with your forehead, like archy the cockroach,” Mutt says","name":"Mutt"},{"index":1929,"str":"A few more people unknown to Mutt and Jeff follow with more wheelbarrows","name":"Mutt"},{"index":2188,"str":"“You’ll need seeds,” Mutt points out","name":"Mutt"},{"index":2675,"str":"“We didn’t know where it was,” Mutt confesses","name":"Mutt"},{"index":4263,"str":"Mutt and Jeff nod","name":"Mutt"},{"index":4743,"str":"Mutt and Jeff shrug","name":"Mutt"},{"index":4764,"str":"“I can’t believe she wants to do it,” Mutt says","name":"Mutt"},{"index":5534,"str":"“And student loans?” Mutt inquires","name":"Mutt"},{"index":5992,"str":"Mutt moves past him and stops by the south railing","name":"Mutt"},{"index":6229,"str":"” Mutt regards Jeff moving slabs of wood into position on a long worktable","name":"Mutt"},{"index":7531,"str":"“You’re the one who made me read him,” objects Mutt","name":"Mutt"},{"index":1929,"str":"A few more people unknown to Mutt and Jeff follow with more wheelbarrows","name":"Jeff"},{"index":2004,"str":"Jeff says, “Here, this box is ready","name":"Jeff"},{"index":3669,"str":"“It seems kind of breezy up here now,” Jeff remarks to him","name":"Jeff"},{"index":4263,"str":"Mutt and Jeff nod","name":"Jeff"},{"index":4743,"str":"Mutt and Jeff shrug","name":"Jeff"},{"index":4865,"str":"“Somebody’s got to do it,” Jeff pontificates","name":"Jeff"},{"index":5160,"str":"“It’s working,” Jeff says","name":"Jeff"},{"index":5866,"str":"” Jeff stands in the open doorway of it, looking south at Wall Street","name":"Jeff"},{"index":6229,"str":"” Mutt regards Jeff moving slabs of wood into position on a long worktable","name":"Jeff"},{"index":6582,"str":"Jeff snorts as he lines up two slabs","name":"Jeff"},{"index":6675,"str":"Jeff shrugs and tries to do it himself","name":"Jeff"},{"index":7051,"str":"“I do,” Jeff admits","name":"Jeff"},{"index":7764,"str":"” Jeff snorts with amusement","name":"Jeff"},{"index":5204,"str":"It’s like that Franklin says, the only problem is if it works so well it wipes out civilization","name":"Franklin"},{"index":1691,"str":"Vlade appears out of the service elevator, pushing a wheelbarrow of black dirt","name":"Vlade"},{"index":2042,"str":"Vlade helps his team fill the new box with soil","name":"Vlade"},{"index":2918,"str":"Vlade is as good as his word, so when the current load of dirt is shoveled into the new planter boxes, he goes to the storage room and pulls out what looks like an oversized suitcase","name":"Vlade"},{"index":3582,"str":"Vlade takes a look around and declares the place rebuilt","name":"Vlade"},{"index":3823,"str":"“Sure,” says Vlade","name":"Vlade"},{"index":4389,"str":"Or maybe you can take Charlotte’s room","name":"Charlotte"}]},{"chapter":118,"name":"","length":469,"links":[]},{"chapter":119,"name":"Stefan and Roberto","length":15219,"links":[{"index":2628,"str":"I bet Inspector Gen would bring home a couple for us","name":"Gen"},{"index":2195,"str":"But it was true that Franklin and Vlade and Charlotte had melted their gold coins and were taking care of the money the gold had been bought for","name":"Franklin"},{"index":2340,"str":"And Franklin in particular was insistent that from now on when they went out to do things, they had to take their wristpad with them, always, no exceptions","name":"Franklin"},{"index":3386,"str":"Franklin shook his head","name":"Franklin"},{"index":3914,"str":"Franklin nodded at this","name":"Franklin"},{"index":4237,"str":"“No no,” Franklin said","name":"Franklin"},{"index":5404,"str":"Today, with Franklin Garr off to join the Cloisterclusterfuck, as he called it, they used the song to convince Mr","name":"Franklin"},{"index":7764,"str":"“Hedge money? Like Franklin Garr?”","name":"Franklin"},{"index":2195,"str":"But it was true that Franklin and Vlade and Charlotte had melted their gold coins and were taking care of the money the gold had been bought for","name":"Vlade"},{"index":4307,"str":"We’ve got it in Vlade’s safe so you don’t make a big necklace out of it and then go swimming while you’re wearing it","name":"Vlade"},{"index":1166,"str":"Hexter wanted Amelia Black to give them a ride so that they would be able to see how much the land looked like the map of it when you were up at the bird’s level","name":"Amelia"},{"index":147,"str":"Whatever this project turned out to be, it was going to be complicated by the fact that now they had about a dozen adults in the Met paying attention to them and bringing up the foster parent thing, the guardian thing, the paper thing, the gold thing, trying to make them “wards of the co-op,” as Charlotte put it at one point when they refused all supervision","name":"Charlotte"},{"index":2195,"str":"But it was true that Franklin and Vlade and Charlotte had melted their gold coins and were taking care of the money the gold had been bought for","name":"Charlotte"},{"index":3526,"str":"Charlotte made that your legal status","name":"Charlotte"},{"index":0,"str":"It took about a week for Stefan and Roberto to eat their way back to weight, and after that Roberto got restless and began to plot their next move","name":"Stefan"},{"index":1887,"str":"“I wonder if all of school would have been this easy?” Stefan said","name":"Stefan"},{"index":3423,"str":"And you, Stefan?”","name":"Stefan"},{"index":3442,"str":"“I am Stefan Melville de Madison","name":"Stefan"},{"index":3622,"str":"“All right already,” Stefan conceded","name":"Stefan"},{"index":7395,"str":"“Why didn’t they take over the uptown towers?” Stefan asked the old man","name":"Stefan"},{"index":13929,"str":"“That might be fun,” Stefan guessed","name":"Stefan"},{"index":14189,"str":"“We could map the city!” Stefan suggested","name":"Stefan"},{"index":0,"str":"It took about a week for Stefan and Roberto to eat their way back to weight, and after that Roberto got restless and began to plot their next move","name":"Roberto"},{"index":2796,"str":"“No to that,” Roberto said","name":"Roberto"},{"index":2966,"str":"In fact, Roberto, how did you get any name at all, being orphaned at birth and self-raised from out of a lobster trap?”","name":"Roberto"},{"index":3087,"str":"Roberto got his stubborn look","name":"Roberto"},{"index":3117,"str":"“I am Roberto New York, of the house of New York","name":"Roberto"},{"index":3167,"str":"The dockmaster called me little robber, so I figured my name was Robber, and then later a guy told me about Roberto Clemente","name":"Roberto"},{"index":3293,"str":"So I decided I was Roberto","name":"Roberto"},{"index":3659,"str":"“We can always zap it later,” he explained to Roberto, forestalling Roberto’s expostulations","name":"Roberto"},{"index":4103,"str":"“Besides, what about our gold,” Roberto demanded","name":"Roberto"},{"index":7501,"str":"“So what?” Roberto said","name":"Roberto"},{"index":8565,"str":"“So that would be their he-ge-mony,” Roberto said carefully","name":"Roberto"},{"index":11782,"str":"Roberto killed the motor and they drifted with the slow flushing of water in this part of the intertidal","name":"Roberto"},{"index":15155,"str":"Roberto wasn’t satisfied","name":"Roberto"}]},{"chapter":120,"name":"","length":708,"links":[]},{"chapter":121,"name":"Charlotte","length":29580,"links":[{"index":14313,"str":"“You know the people who had Mutt and Jeff kidnapped?” Gen said from Charlotte’s wrist","name":"Mutt"},{"index":14313,"str":"“You know the people who had Mutt and Jeff kidnapped?” Gen said from Charlotte’s wrist","name":"Jeff"},{"index":19532,"str":"This was what Jeff had taught her up on the farm, and what he had earlier tried foolishly to correct, or at least express, with his graffiti hacks","name":"Jeff"},{"index":14145,"str":"So she was getting ready to go to dinner with Larry on Tuesday, cutting short any number of other critical items on her to-do list, when Gen Octaviasdottir pinged her","name":"Gen"},{"index":14313,"str":"“You know the people who had Mutt and Jeff kidnapped?” Gen said from Charlotte’s wrist","name":"Gen"},{"index":15213,"str":"“Well, but it’s more complicated than that,” Gen told her as her vision came back","name":"Gen"},{"index":22058,"str":"Anyway, even in 2008 they nationalized General Motors, and they could have nationalized the banks too, as a condition for giving them about fifteen trillion dollars","name":"Gen"},{"index":27694,"str":"I tell the story because it was told to me by a friend of mine, Inspector Gen Octaviasdottir","name":"Gen"},{"index":9933,"str":"More satisfying than her daily grind, or the campaign hammer-and-tongs, was working with Franklin Garr on his redevelopment project","name":"Franklin"},{"index":10159,"str":"Franklin’s investment group (which included the Met gold gang) had secured provisional property rights, as good as could be gotten for the intertidal, plus demolition permits, building permits, and the funding to build","name":"Franklin"},{"index":10736,"str":"Workers in the building trades from Boston to Atlanta were streaming in to New York to reconstruct the city, but there still weren’t enough of them, so the main coup for Franklin had been the assembling of the construction team itself","name":"Franklin"},{"index":11347,"str":"Bedrock in Franklin’s neighborhood turned out to be 160 feet below the canal bottom; this was bad, but not impossibly so","name":"Franklin"},{"index":11483,"str":"The moving parts, or stretchy parts, that would extend between the deep bollards and the floating platforms were the crux of the problem, according to Franklin’s head contractor","name":"Franklin"},{"index":12654,"str":"While gobbling down a quick meal in the deli occupying the prow of the Flatiron, Charlotte asked Franklin what he thought of the situation in finance","name":"Franklin"},{"index":12805,"str":"Franklin waggled a hand as he swallowed and then said, “It’s all happening","name":"Franklin"},{"index":22429,"str":"” Franklin had suggested this riposte","name":"Franklin"},{"index":12138,"str":"Vlade’s friend Idelba is part of that team; she’s dredging the bottom to get things clear before they caisson it and drill to bedrock","name":"Vlade"},{"index":5505,"str":"Also Amelia put out a photo of herself and a leopard under the banner ALL THE GREAT MAMMALS ARE VOTING FOR CHARLOTTE","name":"Amelia"},{"index":5622,"str":"Leopard sitting on its haunches like a dog, Amelia standing right next to it, both of them unclothed and unrepentantly beautiful","name":"Amelia"},{"index":16605,"str":"“Did you ask your friend Amelia Black to start it?”","name":"Amelia"},{"index":17170,"str":"Amelia is maybe a little like that","name":"Amelia"},{"index":17657,"str":"“I only mean,” she said, “that Amelia’s carefully disguised and possibly unconscious brilliance is not the point here","name":"Amelia"},{"index":24655,"str":"“Well, it’s Amelia who started that, but yeah","name":"Amelia"},{"index":0,"str":"Charlotte was running for Congress without wasting too much time on it","name":"Charlotte"},{"index":5861,"str":"“Okay,” Charlotte said","name":"Charlotte"},{"index":7926,"str":"Charlotte paused to wonder what it meant when a police state was aspirational, a staving off of a worse fate, but then she went back to work","name":"Charlotte"},{"index":8101,"str":"What this meant for Charlotte was a constant stream of clients beseeching help to find housing, as their old accommodations had been damaged or wrecked","name":"Charlotte"},{"index":8837,"str":"This gave Charlotte some leverage within the city system and also provided an indirect way to critique the mayor and her people","name":"Charlotte"},{"index":9076,"str":"Charlotte was one node in that alternate system, and without criticizing the mayor outright, she was happy to see a kind of dual power alternative networking below the level of the mayor’s office, which was still focusing most of its efforts on burnishing the mayor’s image, as always","name":"Charlotte"},{"index":9518,"str":"“Who cares what the figurehead on the ship looks like when there’s leaks under the waterline?” Charlotte said in one of her wrist messages to the public","name":"Charlotte"},{"index":12654,"str":"While gobbling down a quick meal in the deli occupying the prow of the Flatiron, Charlotte asked Franklin what he thought of the situation in finance","name":"Charlotte"},{"index":13790,"str":"Falling asleep that night, Charlotte thought over their conversation, and in the morning she sent Larry a message","name":"Charlotte"},{"index":14313,"str":"“You know the people who had Mutt and Jeff kidnapped?” Gen said from Charlotte’s wrist","name":"Charlotte"},{"index":14974,"str":"” Charlotte tried to comprehend","name":"Charlotte"},{"index":15565,"str":"Charlotte thought it was a little strange Larry had suggested it, but she liked the food and didn’t want to be muddying the waters with any countersuggestions, given how busy he must be","name":"Charlotte"},{"index":16352,"str":"“So how’s it going?” Charlotte asked when their drinks had come","name":"Charlotte"},{"index":17390,"str":"Charlotte reminded herself to curb her tongue","name":"Charlotte"},{"index":19998,"str":"“You would never beat anyone,” Charlotte said","name":"Charlotte"},{"index":25223,"str":"At that point Charlotte decided the atmosphere was friendly enough to bring up a delicate matter","name":"Charlotte"},{"index":28903,"str":"“I hope so,” Charlotte said","name":"Charlotte"}]},{"chapter":122,"name":"","length":543,"links":[]},{"chapter":123,"name":"Vlade","length":23404,"links":[{"index":18829,"str":"After another song, standing out under the prow of the Flatiron, Vlade and Idelba watched as Franklin’s friend Jojo approached Charlotte and congratulated her","name":"Franklin"},{"index":18988,"str":"Charlotte thanked her and then called Franklin over and asked them to discuss how they could coordinate their Soho and Chelsea redevelopment projects, such that they combined strengths and both got better","name":"Franklin"},{"index":19194,"str":"Franklin and Jojo agreed to this with a handshake and went over to the drinks table to see if they could find an unopened bottle of bubbly","name":"Franklin"},{"index":20515,"str":"Franklin Garr came breezing in, saw them and came over and leaned down to give Charlotte a hug and a kiss on the head","name":"Franklin"},{"index":0,"str":"Vlade spent his days working on the building, as always","name":"Vlade"},{"index":1396,"str":"And now Vlade was thinking about storm windows for the farm, and that wasn’t going to be easy, or fast, or cheap","name":"Vlade"},{"index":1958,"str":"But not Vlade; even at only ten minutes a day she was better than any of the other board members, as far as he was concerned","name":"Vlade"},{"index":2668,"str":"“I like the sound of that,” Vlade said","name":"Vlade"},{"index":3023,"str":"“Hurray for the storm,” Vlade said, unamused by his own joke","name":"Vlade"},{"index":4409,"str":"Vlade had felt blamed, and tried not to resent it","name":"Vlade"},{"index":7785,"str":"Vlade went with her, towing his boat behind the barge, so he could get back to the city the next day","name":"Vlade"},{"index":8745,"str":"“Inland,” Vlade supposed","name":"Vlade"},{"index":10258,"str":"Vlade didn’t want to put his feet down flat, fins or no; seemed like a curl of broken glass could slice your foot in half, as had happened to Vlade’s brother once when they were kids","name":"Vlade"},{"index":10442,"str":"So Vlade swam horizontally above the bottom, floating around to inspect it","name":"Vlade"},{"index":11789,"str":"The others then played cards, while Vlade and Idelba went out to look at the waterfront, leaning on the rail","name":"Vlade"},{"index":11899,"str":"“What are you going to do?” Vlade asked her, not knowing how else to put it","name":"Vlade"},{"index":14701,"str":"Under the blue waves Vlade now looked at, that Hudson outbreak flood had left a submarine canyon that still cut the continental shelf","name":"Vlade"},{"index":14836,"str":"Vlade had dived its walls in his youth","name":"Vlade"},{"index":15520,"str":"She helped Vlade get the common room ready, and of course for a thing like this there was no shortage of help","name":"Vlade"},{"index":18829,"str":"After another song, standing out under the prow of the Flatiron, Vlade and Idelba watched as Franklin’s friend Jojo approached Charlotte and congratulated her","name":"Vlade"},{"index":19335,"str":"Vlade stood in front of Ettore’s band, swaying to a milonga, feeling the outbreak flood pour through him","name":"Vlade"},{"index":19497,"str":"When the band had played its last song, Vlade walked over to the Met with Charlotte, steering her over the looser gangplanks; she seemed wasted","name":"Vlade"},{"index":19722,"str":"Vlade sat across from them","name":"Vlade"},{"index":19751,"str":"“Maybe you can settle Stefan and Roberto in my room,” she said to Vlade","name":"Vlade"},{"index":20092,"str":"Vlade put a hand to her arm","name":"Vlade"},{"index":20364,"str":"Vlade didn’t get it","name":"Vlade"},{"index":21159,"str":"She looked up at him, and Vlade saw a new look on her face, an idea she liked","name":"Vlade"},{"index":21477,"str":"He was startled, Vlade could see","name":"Vlade"},{"index":21545,"str":"He looked at Vlade: “How far is it?”","name":"Vlade"},{"index":21583,"str":"“You got me,” Vlade said","name":"Vlade"},{"index":22183,"str":"Charlotte looked at Vlade","name":"Vlade"},{"index":22264,"str":"“Really?” Vlade said","name":"Vlade"},{"index":22869,"str":"When they had collected themselves, Charlotte said to Vlade, “So what about Idelba, where is she?”","name":"Vlade"},{"index":23072,"str":"Vlade shrugged","name":"Vlade"},{"index":19643,"str":"In the dining room she sat down heavily next to Amelia Black and Gordon Hexter","name":"Amelia"},{"index":22677,"str":"“You could always get a ride with Amelia on her blimp","name":"Amelia"},{"index":22765,"str":"They all laughed together, even Amelia","name":"Amelia"},{"index":333,"str":"Also the building was still stuffed with new refugees Charlotte insisted had to be sheltered until other arrangements could be found, and this was bad, because the building had already been full before the storm, so now things were simply desperate, and yet at the same time many of their refugees were very grateful to be there and had fallen in love with the Met, the way a limpet falls in love with a pier piling it runs into after being scraped off a ship","name":"Charlotte"},{"index":794,"str":"They would have to be pried off the walls here too, and at that point, as Charlotte had put it, the communal ethos of one for all and all for one would become an obstacle to good governance","name":"Charlotte"},{"index":1766,"str":"He mentioned this to Charlotte once when she was home and they were dealing with refugee guest problems","name":"Charlotte"},{"index":15391,"str":"Then election day came and Charlotte won","name":"Charlotte"},{"index":16388,"str":"Charlotte herself didn’t show up until midnight, having gone to work that day as if it were a normal day","name":"Charlotte"},{"index":18829,"str":"After another song, standing out under the prow of the Flatiron, Vlade and Idelba watched as Franklin’s friend Jojo approached Charlotte and congratulated her","name":"Charlotte"},{"index":18988,"str":"Charlotte thanked her and then called Franklin over and asked them to discuss how they could coordinate their Soho and Chelsea redevelopment projects, such that they combined strengths and both got better","name":"Charlotte"},{"index":19497,"str":"When the band had played its last song, Vlade walked over to the Met with Charlotte, steering her over the looser gangplanks; she seemed wasted","name":"Charlotte"},{"index":20515,"str":"Franklin Garr came breezing in, saw them and came over and leaned down to give Charlotte a hug and a kiss on the head","name":"Charlotte"},{"index":21754,"str":"But yeah,” he said to Charlotte, “of course! Love to take you down there to your coronation","name":"Charlotte"},{"index":22183,"str":"Charlotte looked at Vlade","name":"Charlotte"},{"index":22286,"str":"Charlotte laughed","name":"Charlotte"},{"index":22869,"str":"When they had collected themselves, Charlotte said to Vlade, “So what about Idelba, where is she?”","name":"Charlotte"},{"index":19751,"str":"“Maybe you can settle Stefan and Roberto in my room,” she said to Vlade","name":"Stefan"},{"index":19751,"str":"“Maybe you can settle Stefan and Roberto in my room,” she said to Vlade","name":"Roberto"}]},{"chapter":124,"name":"","length":596,"links":[]},{"chapter":125,"name":"Franklin","length":24631,"links":[{"index":1258,"str":"“So did you hear that Inspector Gen’s data hound cracked Morningside and found out who was making those offers on our building?”","name":"Gen"},{"index":9557,"str":"But on the other hand take Amelia Black, the star of the Met and the cloud and the world; she was over-the-top, not just neat but compelling, not just perfect but interesting; and because for years she had had a professional and/or personal propensity for getting naked on her show, I had not been able to avoid noticing along with the rest of humanity that she also had a spectacular figure, with the extra splashes on a big rangy frame that certainly made for at least part of her popularity, that and her goofy sweet character","name":"Amelia"},{"index":0,"str":"So it had got to the point where I looked up Charlotte Armstrong in the cloud and found out that she was sixteen years older than me","name":"Charlotte"},{"index":7064,"str":"But now I had to get my Charlotte down to D","name":"Charlotte"},{"index":8261,"str":"Our supposed reconciliation and business cooperation pact, enacted at Charlotte’s behest, meant nothing to her; this was what her refusal to look my way conveyed","name":"Charlotte"},{"index":8584,"str":"And then Charlotte appeared, walking onto the marina dock, punctual as usual, weighed down by two fat shoulder bags and clumping along with that little limp she has","name":"Charlotte"},{"index":10750,"str":"Charlotte Armstrong was not perfect, or nice","name":"Charlotte"},{"index":11244,"str":"I angled what I did to please Charlotte Armstrong, to make her laugh","name":"Charlotte"},{"index":12034,"str":"And yes, Charlotte laughed","name":"Charlotte"},{"index":12721,"str":"I had no idea if Charlotte was sensitive to the swell or not, but the last thing I wanted now was for her to get seasick","name":"Charlotte"},{"index":14036,"str":"Charlotte got a call and took it","name":"Charlotte"},{"index":19956,"str":"Once the battery’s charger was plugged in, Charlotte and I walked up to a restaurant with its windows overlooking the marina","name":"Charlotte"},{"index":23088,"str":"But Charlotte Armstrong was sexy","name":"Charlotte"}]},{"chapter":126,"name":"","length":179,"links":[]},{"chapter":127,"name":"Amelia","length":14267,"links":[{"index":2913,"str":"I got used to talking to people again from doing it up here in the cloud, and then one time I came through New York, and the mooring at the Met was available, and I met with Vlade, up in the cupola, and I liked him","name":"Vlade"},{"index":3227,"str":"“Does Vlade know the part he played in reeling you in?” Mr","name":"Vlade"},{"index":87,"str":"Amelia took Stefan and Roberto and Mr","name":"Amelia"},{"index":514,"str":"“I see them all the time,” Amelia said","name":"Amelia"},{"index":645,"str":"Hexter toured the gondola, with Amelia explaining everything, including the claw marks made by her polar bears, which she could now point at with only a brief plunge into sadness","name":"Amelia"},{"index":1193,"str":"Category: Amelia’s Dumb Moves, Extrication From","name":"Amelia"},{"index":1400,"str":"They gathered in the glass-bottomed bow of the gondola and looked down at the city as they ate tofu burgers that Amelia had prepared on her kitchen stove","name":"Amelia"},{"index":1623,"str":"“I think it’s a million now,” Amelia said","name":"Amelia"},{"index":2529,"str":"Amelia nodded, feeling herself blush","name":"Amelia"},{"index":4102,"str":"“I don’t think orcas come in the harbor,” Amelia said","name":"Amelia"},{"index":5095,"str":"“Where’s your marsh?” Amelia asked","name":"Amelia"},{"index":5944,"str":"“So is it true that these towers are mostly occupied now?” Amelia asked, diverting them from what looked to be an ongoing dispute","name":"Amelia"},{"index":6767,"str":"“I like the clean lines,” Amelia said","name":"Amelia"},{"index":6977,"str":"“Can there really be enough apartments for all those people?” Amelia asked","name":"Amelia"},{"index":7377,"str":"Amelia shook her head","name":"Amelia"},{"index":8131,"str":"Amelia asked Frans to head south just offshore from Manhattan, and they regarded the city in silence as they turned and floated by it","name":"Amelia"},{"index":12934,"str":"“Isn’t that where you used to live?” Amelia inquired, pointing down","name":"Amelia"},{"index":13617,"str":"“What about you guys?” Amelia asked","name":"Amelia"},{"index":14042,"str":"“Well good,” Amelia said","name":"Amelia"},{"index":13729,"str":"“Charlotte wants us to house-sit her room while she’s in Washington","name":"Charlotte"},{"index":87,"str":"Amelia took Stefan and Roberto and Mr","name":"Stefan"},{"index":5699,"str":"“I thought it was the Institute of Stefan and Roberto,” Hexter said","name":"Stefan"},{"index":5876,"str":"“That’s because animals always have homes,” Stefan explained again","name":"Stefan"},{"index":13571,"str":"“We like having you around,” Stefan ventured","name":"Stefan"},{"index":14015,"str":"“Me neither,” Stefan said","name":"Stefan"},{"index":14194,"str":"Stefan and Roberto and Mr","name":"Stefan"},{"index":87,"str":"Amelia took Stefan and Roberto and Mr","name":"Roberto"},{"index":2567,"str":"“Why?” Roberto asked","name":"Roberto"},{"index":3445,"str":"“We see you,” Roberto declared","name":"Roberto"},{"index":5132,"str":"“There by that skinny tall building,” Roberto said","name":"Roberto"},{"index":5699,"str":"“I thought it was the Institute of Stefan and Roberto,” Hexter said","name":"Roberto"},{"index":8611,"str":"“What happened?” Roberto asked","name":"Roberto"},{"index":10450,"str":"“Cool,” Roberto said","name":"Roberto"},{"index":10825,"str":"“So what the hell?” Roberto said","name":"Roberto"},{"index":11704,"str":"“I don’t get it,” said Roberto","name":"Roberto"},{"index":12171,"str":"Roberto supposed it might be better just to win outright","name":"Roberto"},{"index":13693,"str":"“Don’t know,” Roberto said uneasily","name":"Roberto"},{"index":14194,"str":"Stefan and Roberto and Mr","name":"Roberto"}]},{"chapter":128,"name":"","length":290,"links":[]},{"chapter":129,"name":"Citizen","length":9164,"links":[{"index":5151,"str":"Note that this flurry of social and legal change did not happen because of Representative Charlotte Armstrong of the Twelfth District of the State of New York, also known as “Red Charlotte,” admirable woman and congressperson though she was","name":"Charlotte"}]},{"chapter":130,"name":"","length":428,"links":[]},{"chapter":131,"name":"Mutt and Jeff","length":14654,"links":[{"index":0,"str":"Later that year, in the depths of winter, Mutt and Jeff go downstairs from their hotello on the farm floor, where they have stubbornly remained despite the fact that a hotello is very difficult to heat properly","name":"Mutt"},{"index":584,"str":"There’s a big crowd in the common room, and Mutt and Jeff sit against a wall watching the action and behaving like the wallflowers they are","name":"Mutt"},{"index":791,"str":"Mutt agrees; Jeff squints","name":"Mutt"},{"index":1103,"str":"“So those two are a couple now?” Jeff asks Mutt","name":"Mutt"},{"index":1495,"str":"Jeff grumbles some kind of objection to this characterization, but Mutt is having none of it","name":"Mutt"},{"index":2627,"str":"She sits down wearily beside Mutt and Jeff and Mr","name":"Mutt"},{"index":2793,"str":"“It’s nice to be outside,” Mutt explains","name":"Mutt"},{"index":3323,"str":"“What are you going to do with all the money coming in?” Mutt asks her","name":"Mutt"},{"index":3989,"str":"“I hope that’s true,” Mutt says","name":"Mutt"},{"index":4283,"str":"“Hey Inspector!” Mutt says","name":"Mutt"},{"index":4737,"str":"“Hey speaking of cases,” Mutt says, “did you ever find out anything more about whoever it was who kept us in that container?”","name":"Mutt"},{"index":5650,"str":"“We hope so,” Mutt says","name":"Mutt"},{"index":6045,"str":"“Not me,” Mutt says","name":"Mutt"},{"index":6546,"str":"Mutt and Jeff know what that’s like, and give each other a glance","name":"Mutt"},{"index":7086,"str":"Then, just as the Institute of Mutt and Jeff is about to call it a night and retire to their hotello, Amelia Black breezes by and grabs them by the arm","name":"Mutt"},{"index":10132,"str":"Some are born bad, some achieve badness … Mutt, the situation having been thrust upon him, moves in tiny abrupt jerks","name":"Mutt"},{"index":10455,"str":"Jeff yells in Mutt’s ear, “Our gal is a terrible dancer!”","name":"Mutt"},{"index":12624,"str":"Mutt can’t stop laughing at the sight of his two friends’ gyrations","name":"Mutt"},{"index":0,"str":"Later that year, in the depths of winter, Mutt and Jeff go downstairs from their hotello on the farm floor, where they have stubbornly remained despite the fact that a hotello is very difficult to heat properly","name":"Jeff"},{"index":584,"str":"There’s a big crowd in the common room, and Mutt and Jeff sit against a wall watching the action and behaving like the wallflowers they are","name":"Jeff"},{"index":791,"str":"Mutt agrees; Jeff squints","name":"Jeff"},{"index":1103,"str":"“So those two are a couple now?” Jeff asks Mutt","name":"Jeff"},{"index":1495,"str":"Jeff grumbles some kind of objection to this characterization, but Mutt is having none of it","name":"Jeff"},{"index":1589,"str":"“Come on, Jeff","name":"Jeff"},{"index":2627,"str":"She sits down wearily beside Mutt and Jeff and Mr","name":"Jeff"},{"index":2902,"str":"“Like forever,” Jeff adds","name":"Jeff"},{"index":5092,"str":"“Or that they got away with it,” Jeff adds grimly","name":"Jeff"},{"index":5338,"str":"“I wondered about that,” Jeff says","name":"Jeff"},{"index":6546,"str":"Mutt and Jeff know what that’s like, and give each other a glance","name":"Jeff"},{"index":7086,"str":"Then, just as the Institute of Mutt and Jeff is about to call it a night and retire to their hotello, Amelia Black breezes by and grabs them by the arm","name":"Jeff"},{"index":7361,"str":"“Can’t you hire escorts?” Jeff asks grumpily","name":"Jeff"},{"index":10251,"str":"Jeff flails so spastically he achieves some kind of nerd sublime","name":"Jeff"},{"index":10455,"str":"Jeff yells in Mutt’s ear, “Our gal is a terrible dancer!”","name":"Jeff"},{"index":12421,"str":"Jeff is a dancing fool; there are so many rhythms in this music that he almost matches one","name":"Jeff"},{"index":4224,"str":"Then Inspector Gen emerges from the elevator and walks by","name":"Gen"},{"index":5675,"str":"Inspector Gen nods wearily","name":"Gen"},{"index":917,"str":"And in fact she comes out of the elevator that very moment, with Franklin Garr","name":"Franklin"},{"index":7086,"str":"Then, just as the Institute of Mutt and Jeff is about to call it a night and retire to their hotello, Amelia Black breezes by and grabs them by the arm","name":"Amelia"},{"index":7407,"str":"Amelia pretends to be offended","name":"Amelia"},{"index":8030,"str":"Amelia steers them up a few blocks and then hangs a right on Thirty-third","name":"Amelia"},{"index":8357,"str":"A door with MEZZROW’S painted on it opens its Judas window, and Amelia puts her face on view","name":"Amelia"},{"index":8665,"str":"Squeezing behind these people, Amelia leads the guys to the back, where there is another door, and a doorman taking a fee for entering","name":"Amelia"},{"index":8801,"str":"Amelia shows him her wristpad and they are all three waved in","name":"Amelia"},{"index":9978,"str":"Amelia and the guys have been sitting against the back wall, but now Amelia pulls them to their feet and they join the dancing","name":"Amelia"},{"index":10317,"str":"Amelia, somewhat to their surprise, was just born bad","name":"Amelia"},{"index":10574,"str":"“That’s Amelia for ya","name":"Amelia"},{"index":11167,"str":"“Bass clarinet!” Amelia shouts at them","name":"Amelia"},{"index":12586,"str":"And he is Nureyev compared to Amelia","name":"Amelia"},{"index":12693,"str":"Amelia is grinning at him","name":"Amelia"},{"index":13231,"str":"Many songs later Amelia makes a gesture, and the guys nod","name":"Amelia"},{"index":14015,"str":"They walk out onto the frozen canal, Amelia holding on to the guys, all stepping carefully","name":"Amelia"},{"index":211,"str":"They join a little party welcoming Charlotte back home from D","name":"Charlotte"},{"index":817,"str":"“But where’s Charlotte?”","name":"Charlotte"},{"index":1718,"str":"And now our young comrade here has not only called out the fixes for Charlotte, he also designed the crash that allowed the key to start turning","name":"Charlotte"},{"index":1943,"str":"“But Charlotte is kind of a shark too","name":"Charlotte"},{"index":2514,"str":"Charlotte regards them","name":"Charlotte"},{"index":3171,"str":"“Good to hear,” Charlotte said","name":"Charlotte"},{"index":4097,"str":"There’s only one Charlotte","name":"Charlotte"},{"index":2292,"str":"Stefan and Roberto are running around serving drinks to people, and it’s looking like they have filched a few too many sips, as they are glassy-eyed and perhaps might have to do like Romans and go spew and then carry on","name":"Stefan"},{"index":2292,"str":"Stefan and Roberto are running around serving drinks to people, and it’s looking like they have filched a few too many sips, as they are glassy-eyed and perhaps might have to do like Romans and go spew and then carry on","name":"Roberto"}]}]
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
typeof define === 'function' && define.amd ? define(['exports'], factory) :
(factory((global.d3 = global.d3 || {})));
}(this, function (exports) { 'use strict';
function ascending(a, b) {
return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
}
function bisector(compare) {
if (compare.length === 1) compare = ascendingComparator(compare);
return {
left: function(a, x, lo, hi) {
if (lo == null) lo = 0;
if (hi == null) hi = a.length;
while (lo < hi) {
var mid = lo + hi >>> 1;
if (compare(a[mid], x) < 0) lo = mid + 1;
else hi = mid;
}
return lo;
},
right: function(a, x, lo, hi) {
if (lo == null) lo = 0;
if (hi == null) hi = a.length;
while (lo < hi) {
var mid = lo + hi >>> 1;
if (compare(a[mid], x) > 0) hi = mid;
else lo = mid + 1;
}
return lo;
}
};
}
function ascendingComparator(f) {
return function(d, x) {
return ascending(f(d), x);
};
}
var ascendingBisect = bisector(ascending);
var bisectRight = ascendingBisect.right;
var bisectLeft = ascendingBisect.left;
function descending(a, b) {
return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
}
function number(x) {
return x === null ? NaN : +x;
}
function variance(array, f) {
var n = array.length,
m = 0,
a,
d,
s = 0,
i = -1,
j = 0;
if (f == null) {
while (++i < n) {
if (!isNaN(a = number(array[i]))) {
d = a - m;
m += d / ++j;
s += d * (a - m);
}
}
}
else {
while (++i < n) {
if (!isNaN(a = number(f(array[i], i, array)))) {
d = a - m;
m += d / ++j;
s += d * (a - m);
}
}
}
if (j > 1) return s / (j - 1);
}
function deviation(array, f) {
var v = variance(array, f);
return v ? Math.sqrt(v) : v;
}
function extent(array, f) {
var i = -1,
n = array.length,
a,
b,
c;
if (f == null) {
while (++i < n) if ((b = array[i]) != null && b >= b) { a = c = b; break; }
while (++i < n) if ((b = array[i]) != null) {
if (a > b) a = b;
if (c < b) c = b;
}
}
else {
while (++i < n) if ((b = f(array[i], i, array)) != null && b >= b) { a = c = b; break; }
while (++i < n) if ((b = f(array[i], i, array)) != null) {
if (a > b) a = b;
if (c < b) c = b;
}
}
return [a, c];
}
var array = Array.prototype;
var slice = array.slice;
var map = array.map;
function constant(x) {
return function() {
return x;
};
}
function identity(x) {
return x;
}
function range(start, stop, step) {
start = +start, stop = +stop, step = (n = arguments.length) < 2 ? (stop = start, start = 0, 1) : n < 3 ? 1 : +step;
var i = -1,
n = Math.max(0, Math.ceil((stop - start) / step)) | 0,
range = new Array(n);
while (++i < n) {
range[i] = start + i * step;
}
return range;
}
var e10 = Math.sqrt(50);
var e5 = Math.sqrt(10);
var e2 = Math.sqrt(2);
function ticks(start, stop, count) {
var step = tickStep(start, stop, count);
return range(
Math.ceil(start / step) * step,
Math.floor(stop / step) * step + step / 2, // inclusive
step
);
}
function tickStep(start, stop, count) {
var step0 = Math.abs(stop - start) / Math.max(0, count),
step1 = Math.pow(10, Math.floor(Math.log(step0) / Math.LN10)),
error = step0 / step1;
if (error >= e10) step1 *= 10;
else if (error >= e5) step1 *= 5;
else if (error >= e2) step1 *= 2;
return stop < start ? -step1 : step1;
}
function sturges(values) {
return Math.ceil(Math.log(values.length) / Math.LN2) + 1;
}
function histogram() {
var value = identity,
domain = extent,
threshold = sturges;
function histogram(data) {
var i,
n = data.length,
x,
values = new Array(n);
for (i = 0; i < n; ++i) {
values[i] = value(data[i], i, data);
}
var xz = domain(values),
x0 = xz[0],
x1 = xz[1],
tz = threshold(values, x0, x1);
// Convert number of thresholds into uniform thresholds.
if (!Array.isArray(tz)) tz = ticks(x0, x1, tz);
// Remove any thresholds outside the domain.
var m = tz.length;
while (tz[0] <= x0) tz.shift(), --m;
while (tz[m - 1] >= x1) tz.pop(), --m;
var bins = new Array(m + 1),
bin;
// Initialize bins.
for (i = 0; i <= m; ++i) {
bin = bins[i] = [];
bin.x0 = i > 0 ? tz[i - 1] : x0;
bin.x1 = i < m ? tz[i] : x1;
}
// Assign data to bins by value, ignoring any outside the domain.
for (i = 0; i < n; ++i) {
x = values[i];
if (x0 <= x && x <= x1) {
bins[bisectRight(tz, x, 0, m)].push(data[i]);
}
}
return bins;
}
histogram.value = function(_) {
return arguments.length ? (value = typeof _ === "function" ? _ : constant(_), histogram) : value;
};
histogram.domain = function(_) {
return arguments.length ? (domain = typeof _ === "function" ? _ : constant([_[0], _[1]]), histogram) : domain;
};
histogram.thresholds = function(_) {
return arguments.length ? (threshold = typeof _ === "function" ? _ : Array.isArray(_) ? constant(slice.call(_)) : constant(_), histogram) : threshold;
};
return histogram;
}
function threshold(array, p, f) {
if (f == null) f = number;
if (!(n = array.length)) return;
if ((p = +p) <= 0 || n < 2) return +f(array[0], 0, array);
if (p >= 1) return +f(array[n - 1], n - 1, array);
var n,
h = (n - 1) * p,
i = Math.floor(h),
a = +f(array[i], i, array),
b = +f(array[i + 1], i + 1, array);
return a + (b - a) * (h - i);
}
function freedmanDiaconis(values, min, max) {
values = map.call(values, number).sort(ascending);
return Math.ceil((max - min) / (2 * (threshold(values, 0.75) - threshold(values, 0.25)) * Math.pow(values.length, -1 / 3)));
}
function scott(values, min, max) {
return Math.ceil((max - min) / (3.5 * deviation(values) * Math.pow(values.length, -1 / 3)));
}
function max(array, f) {
var i = -1,
n = array.length,
a,
b;
if (f == null) {
while (++i < n) if ((b = array[i]) != null && b >= b) { a = b; break; }
while (++i < n) if ((b = array[i]) != null && b > a) a = b;
}
else {
while (++i < n) if ((b = f(array[i], i, array)) != null && b >= b) { a = b; break; }
while (++i < n) if ((b = f(array[i], i, array)) != null && b > a) a = b;
}
return a;
}
function mean(array, f) {
var s = 0,
n = array.length,
a,
i = -1,
j = n;
if (f == null) {
while (++i < n) if (!isNaN(a = number(array[i]))) s += a; else --j;
}
else {
while (++i < n) if (!isNaN(a = number(f(array[i], i, array)))) s += a; else --j;
}
if (j) return s / j;
}
function median(array, f) {
var numbers = [],
n = array.length,
a,
i = -1;
if (f == null) {
while (++i < n) if (!isNaN(a = number(array[i]))) numbers.push(a);
}
else {
while (++i < n) if (!isNaN(a = number(f(array[i], i, array)))) numbers.push(a);
}
return threshold(numbers.sort(ascending), 0.5);
}
function merge(arrays) {
var n = arrays.length,
m,
i = -1,
j = 0,
merged,
array;
while (++i < n) j += arrays[i].length;
merged = new Array(j);
while (--n >= 0) {
array = arrays[n];
m = array.length;
while (--m >= 0) {
merged[--j] = array[m];
}
}
return merged;
}
function min(array, f) {
var i = -1,
n = array.length,
a,
b;
if (f == null) {
while (++i < n) if ((b = array[i]) != null && b >= b) { a = b; break; }
while (++i < n) if ((b = array[i]) != null && a > b) a = b;
}
else {
while (++i < n) if ((b = f(array[i], i, array)) != null && b >= b) { a = b; break; }
while (++i < n) if ((b = f(array[i], i, array)) != null && a > b) a = b;
}
return a;
}
function pairs(array) {
var i = 0, n = array.length - 1, p = array[0], pairs = new Array(n < 0 ? 0 : n);
while (i < n) pairs[i] = [p, p = array[++i]];
return pairs;
}
function permute(array, indexes) {
var i = indexes.length, permutes = new Array(i);
while (i--) permutes[i] = array[indexes[i]];
return permutes;
}
function scan(array, compare) {
if (!(n = array.length)) return;
var i = 0,
n,
j = 0,
xi,
xj = array[j];
if (!compare) compare = ascending;
while (++i < n) if (compare(xi = array[i], xj) < 0 || compare(xj, xj) !== 0) xj = xi, j = i;
if (compare(xj, xj) === 0) return j;
}
function shuffle(array, i0, i1) {
var m = (i1 == null ? array.length : i1) - (i0 = i0 == null ? 0 : +i0),
t,
i;
while (m) {
i = Math.random() * m-- | 0;
t = array[m + i0];
array[m + i0] = array[i + i0];
array[i + i0] = t;
}
return array;
}
function sum(array, f) {
var s = 0,
n = array.length,
a,
i = -1;
if (f == null) {
while (++i < n) if (a = +array[i]) s += a; // Note: zero and null are equivalent.
}
else {
while (++i < n) if (a = +f(array[i], i, array)) s += a;
}
return s;
}
function transpose(matrix) {
if (!(n = matrix.length)) return [];
for (var i = -1, m = min(matrix, length), transpose = new Array(m); ++i < m;) {
for (var j = -1, n, row = transpose[i] = new Array(n); ++j < n;) {
row[j] = matrix[j][i];
}
}
return transpose;
}
function length(d) {
return d.length;
}
function zip() {
return transpose(arguments);
}
var prefix = "$";
function Map() {}
Map.prototype = map$1.prototype = {
constructor: Map,
has: function(key) {
return (prefix + key) in this;
},
get: function(key) {
return this[prefix + key];
},
set: function(key, value) {
this[prefix + key] = value;
return this;
},
remove: function(key) {
var property = prefix + key;
return property in this && delete this[property];
},
clear: function() {
for (var property in this) if (property[0] === prefix) delete this[property];
},
keys: function() {
var keys = [];
for (var property in this) if (property[0] === prefix) keys.push(property.slice(1));
return keys;
},
values: function() {
var values = [];
for (var property in this) if (property[0] === prefix) values.push(this[property]);
return values;
},
entries: function() {
var entries = [];
for (var property in this) if (property[0] === prefix) entries.push({key: property.slice(1), value: this[property]});
return entries;
},
size: function() {
var size = 0;
for (var property in this) if (property[0] === prefix) ++size;
return size;
},
empty: function() {
for (var property in this) if (property[0] === prefix) return false;
return true;
},
each: function(f) {
for (var property in this) if (property[0] === prefix) f(this[property], property.slice(1), this);
}
};
function map$1(object, f) {
var map = new Map;
// Copy constructor.
if (object instanceof Map) object.each(function(value, key) { map.set(key, value); });
// Index array by numeric index or specified key function.
else if (Array.isArray(object)) {
var i = -1,
n = object.length,
o;
if (f == null) while (++i < n) map.set(i, object[i]);
else while (++i < n) map.set(f(o = object[i], i, object), o);
}
// Convert object to map.
else if (object) for (var key in object) map.set(key, object[key]);
return map;
}
function nest() {
var keys = [],
sortKeys = [],
sortValues,
rollup,
nest;
function apply(array, depth, createResult, setResult) {
if (depth >= keys.length) return rollup != null
? rollup(array) : (sortValues != null
? array.sort(sortValues)
: array);
var i = -1,
n = array.length,
key = keys[depth++],
keyValue,
value,
valuesByKey = map$1(),
values,
result = createResult();
while (++i < n) {
if (values = valuesByKey.get(keyValue = key(value = array[i]) + "")) {
values.push(value);
} else {
valuesByKey.set(keyValue, [value]);
}
}
valuesByKey.each(function(values, key) {
setResult(result, key, apply(values, depth, createResult, setResult));
});
return result;
}
function entries(map, depth) {
if (++depth > keys.length) return map;
var array, sortKey = sortKeys[depth - 1];
if (rollup != null && depth >= keys.length) array = map.entries();
else array = [], map.each(function(v, k) { array.push({key: k, values: entries(v, depth)}); });
return sortKey != null ? array.sort(function(a, b) { return sortKey(a.key, b.key); }) : array;
}
return nest = {
object: function(array) { return apply(array, 0, createObject, setObject); },
map: function(array) { return apply(array, 0, createMap, setMap); },
entries: function(array) { return entries(apply(array, 0, createMap, setMap), 0); },
key: function(d) { keys.push(d); return nest; },
sortKeys: function(order) { sortKeys[keys.length - 1] = order; return nest; },
sortValues: function(order) { sortValues = order; return nest; },
rollup: function(f) { rollup = f; return nest; }
};
}
function createObject() {
return {};
}
function setObject(object, key, value) {
object[key] = value;
}
function createMap() {
return map$1();
}
function setMap(map, key, value) {
map.set(key, value);
}
function Set() {}
var proto = map$1.prototype;
Set.prototype = set.prototype = {
constructor: Set,
has: proto.has,
add: function(value) {
value += "";
this[prefix + value] = value;
return this;
},
remove: proto.remove,
clear: proto.clear,
values: proto.keys,
size: proto.size,
empty: proto.empty,
each: proto.each
};
function set(object, f) {
var set = new Set;
// Copy constructor.
if (object instanceof Set) object.each(function(value) { set.add(value); });
// Otherwise, assume it’s an array.
else if (object) {
var i = -1, n = object.length;
if (f == null) while (++i < n) set.add(object[i]);
else while (++i < n) set.add(f(object[i], i, object));
}
return set;
}
function d3keys(map) {
var keys = [];
for (var key in map) keys.push(key);
return keys;
}
function values(map) {
var values = [];
for (var key in map) values.push(map[key]);
return values;
}
function entries(map) {
var entries = [];
for (var key in map) entries.push({key: key, value: map[key]});
return entries;
}
function uniform(min, max) {
min = min == null ? 0 : +min;
max = max == null ? 1 : +max;
if (arguments.length === 1) max = min, min = 0;
else max -= min;
return function() {
return Math.random() * max + min;
};
}
function normal(mu, sigma) {
var x, r;
mu = mu == null ? 0 : +mu;
sigma = sigma == null ? 1 : +sigma;
return function() {
var y;
// If available, use the second previously-generated uniform random.
if (x != null) y = x, x = null;
// Otherwise, generate a new x and y.
else do {
x = Math.random() * 2 - 1;
y = Math.random() * 2 - 1;
r = x * x + y * y;
} while (!r || r > 1);
return mu + sigma * y * Math.sqrt(-2 * Math.log(r) / r);
};
}
function logNormal() {
var randomNormal = normal.apply(this, arguments);
return function() {
return Math.exp(randomNormal());
};
}
function irwinHall(n) {
return function() {
for (var sum = 0, i = 0; i < n; ++i) sum += Math.random();
return sum;
};
}
function bates(n) {
var randomIrwinHall = irwinHall(n);
return function() {
return randomIrwinHall() / n;
};
}
function exponential(lambda) {
return function() {
return -Math.log(1 - Math.random()) / lambda;
};
}
function linear(t) {
return +t;
}
function quadIn(t) {
return t * t;
}
function quadOut(t) {
return t * (2 - t);
}
function quadInOut(t) {
return ((t *= 2) <= 1 ? t * t : --t * (2 - t) + 1) / 2;
}
function cubicIn(t) {
return t * t * t;
}
function cubicOut(t) {
return --t * t * t + 1;
}
function easeCubicInOut(t) {
return ((t *= 2) <= 1 ? t * t * t : (t -= 2) * t * t + 2) / 2;
}
var exponent = 3;
var polyIn = (function custom(e) {
e = +e;
function polyIn(t) {
return Math.pow(t, e);
}
polyIn.exponent = custom;
return polyIn;
})(exponent);
var polyOut = (function custom(e) {
e = +e;
function polyOut(t) {
return 1 - Math.pow(1 - t, e);
}
polyOut.exponent = custom;
return polyOut;
})(exponent);
var polyInOut = (function custom(e) {
e = +e;
function polyInOut(t) {
return ((t *= 2) <= 1 ? Math.pow(t, e) : 2 - Math.pow(2 - t, e)) / 2;
}
polyInOut.exponent = custom;
return polyInOut;
})(exponent);
var pi = Math.PI;
var halfPi = pi / 2;
function sinIn(t) {
return 1 - Math.cos(t * halfPi);
}
function sinOut(t) {
return Math.sin(t * halfPi);
}
function sinInOut(t) {
return (1 - Math.cos(pi * t)) / 2;
}
function expIn(t) {
return Math.pow(2, 10 * t - 10);
}
function expOut(t) {
return 1 - Math.pow(2, -10 * t);
}
function expInOut(t) {
return ((t *= 2) <= 1 ? Math.pow(2, 10 * t - 10) : 2 - Math.pow(2, 10 - 10 * t)) / 2;
}
function circleIn(t) {
return 1 - Math.sqrt(1 - t * t);
}
function circleOut(t) {
return Math.sqrt(1 - --t * t);
}
function circleInOut(t) {
return ((t *= 2) <= 1 ? 1 - Math.sqrt(1 - t * t) : Math.sqrt(1 - (t -= 2) * t) + 1) / 2;
}
var b1 = 4 / 11;
var b2 = 6 / 11;
var b3 = 8 / 11;
var b4 = 3 / 4;
var b5 = 9 / 11;
var b6 = 10 / 11;
var b7 = 15 / 16;
var b8 = 21 / 22;
var b9 = 63 / 64;
var b0 = 1 / b1 / b1;
function bounceIn(t) {
return 1 - bounceOut(1 - t);
}
function bounceOut(t) {
return (t = +t) < b1 ? b0 * t * t : t < b3 ? b0 * (t -= b2) * t + b4 : t < b6 ? b0 * (t -= b5) * t + b7 : b0 * (t -= b8) * t + b9;
}
function bounceInOut(t) {
return ((t *= 2) <= 1 ? 1 - bounceOut(1 - t) : bounceOut(t - 1) + 1) / 2;
}
var overshoot = 1.70158;
var backIn = (function custom(s) {
s = +s;
function backIn(t) {
return t * t * ((s + 1) * t - s);
}
backIn.overshoot = custom;
return backIn;
})(overshoot);
var backOut = (function custom(s) {
s = +s;
function backOut(t) {
return --t * t * ((s + 1) * t + s) + 1;
}
backOut.overshoot = custom;
return backOut;
})(overshoot);
var backInOut = (function custom(s) {
s = +s;
function backInOut(t) {
return ((t *= 2) < 1 ? t * t * ((s + 1) * t - s) : (t -= 2) * t * ((s + 1) * t + s) + 2) / 2;
}
backInOut.overshoot = custom;
return backInOut;
})(overshoot);
var tau = 2 * Math.PI;
var amplitude = 1;
var period = 0.3;
var elasticIn = (function custom(a, p) {
var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);
function elasticIn(t) {
return a * Math.pow(2, 10 * --t) * Math.sin((s - t) / p);
}
elasticIn.amplitude = function(a) { return custom(a, p * tau); };
elasticIn.period = function(p) { return custom(a, p); };
return elasticIn;
})(amplitude, period);
var elasticOut = (function custom(a, p) {
var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);
function elasticOut(t) {
return 1 - a * Math.pow(2, -10 * (t = +t)) * Math.sin((t + s) / p);
}
elasticOut.amplitude = function(a) { return custom(a, p * tau); };
elasticOut.period = function(p) { return custom(a, p); };
return elasticOut;
})(amplitude, period);
var elasticInOut = (function custom(a, p) {
var s = Math.asin(1 / (a = Math.max(1, a))) * (p /= tau);
function elasticInOut(t) {
return ((t = t * 2 - 1) < 0
? a * Math.pow(2, 10 * t) * Math.sin((s - t) / p)
: 2 - a * Math.pow(2, -10 * t) * Math.sin((s + t) / p)) / 2;
}
elasticInOut.amplitude = function(a) { return custom(a, p * tau); };
elasticInOut.period = function(p) { return custom(a, p); };
return elasticInOut;
})(amplitude, period);
function area(polygon) {
var i = -1,
n = polygon.length,
a,
b = polygon[n - 1],
area = 0;
while (++i < n) {
a = b;
b = polygon[i];
area += a[1] * b[0] - a[0] * b[1];
}
return area / 2;
}
function centroid(polygon) {
var i = -1,
n = polygon.length,
x = 0,
y = 0,
a,
b = polygon[n - 1],
c,
k = 0;
while (++i < n) {
a = b;
b = polygon[i];
k += c = a[0] * b[1] - b[0] * a[1];
x += (a[0] + b[0]) * c;
y += (a[1] + b[1]) * c;
}
return k *= 3, [x / k, y / k];
}
// Returns the 2D cross product of AB and AC vectors, i.e., the z-component of
// the 3D cross product in a quadrant I Cartesian coordinate system (+x is
// right, +y is up). Returns a positive value if ABC is counter-clockwise,
// negative if clockwise, and zero if the points are collinear.
function cross(a, b, c) {
return (b[0] - a[0]) * (c[1] - a[1]) - (b[1] - a[1]) * (c[0] - a[0]);
}
function lexicographicOrder(a, b) {
return a[0] - b[0] || a[1] - b[1];
}
// Computes the upper convex hull per the monotone chain algorithm.
// Assumes points.length >= 3, is sorted by x, unique in y.
// Returns an array of indices into points in left-to-right order.
function computeUpperHullIndexes(points) {
var n = points.length,
indexes = [0, 1],
size = 2;
for (var i = 2; i < n; ++i) {
while (size > 1 && cross(points[indexes[size - 2]], points[indexes[size - 1]], points[i]) <= 0) --size;
indexes[size++] = i;
}
return indexes.slice(0, size); // remove popped points
}
function hull(points) {
if ((n = points.length) < 3) return null;
var i,
n,
sortedPoints = new Array(n),
flippedPoints = new Array(n);
for (i = 0; i < n; ++i) sortedPoints[i] = [+points[i][0], +points[i][1], i];
sortedPoints.sort(lexicographicOrder);
for (i = 0; i < n; ++i) flippedPoints[i] = [sortedPoints[i][0], -sortedPoints[i][1]];
var upperIndexes = computeUpperHullIndexes(sortedPoints),
lowerIndexes = computeUpperHullIndexes(flippedPoints);
// Construct the hull polygon, removing possible duplicate endpoints.
var skipLeft = lowerIndexes[0] === upperIndexes[0],
skipRight = lowerIndexes[lowerIndexes.length - 1] === upperIndexes[upperIndexes.length - 1],
hull = [];
// Add upper hull in right-to-l order.
// Then add lower hull in left-to-right order.
for (i = upperIndexes.length - 1; i >= 0; --i) hull.push(points[sortedPoints[upperIndexes[i]][2]]);
for (i = +skipLeft; i < lowerIndexes.length - skipRight; ++i) hull.push(points[sortedPoints[lowerIndexes[i]][2]]);
return hull;
}
function contains(polygon, point) {
var n = polygon.length,
p = polygon[n - 1],
x = point[0], y = point[1],
x0 = p[0], y0 = p[1],
x1, y1,
inside = false;
for (var i = 0; i < n; ++i) {
p = polygon[i], x1 = p[0], y1 = p[1];
if (((y1 > y) !== (y0 > y)) && (x < (x0 - x1) * (y - y1) / (y0 - y1) + x1)) inside = !inside;
x0 = x1, y0 = y1;
}
return inside;
}
function length$1(polygon) {
var i = -1,
n = polygon.length,
b = polygon[n - 1],
xa,
ya,
xb = b[0],
yb = b[1],
perimeter = 0;
while (++i < n) {
xa = xb;
ya = yb;
b = polygon[i];
xb = b[0];
yb = b[1];
xa -= xb;
ya -= yb;
perimeter += Math.sqrt(xa * xa + ya * ya);
}
return perimeter;
}
var pi$1 = Math.PI;
var tau$1 = 2 * pi$1;
var epsilon = 1e-6;
var tauEpsilon = tau$1 - epsilon;
function Path() {
this._x0 = this._y0 = // start of current subpath
this._x1 = this._y1 = null; // end of current subpath
this._ = [];
}
function path() {
return new Path;
}
Path.prototype = path.prototype = {
constructor: Path,
moveTo: function(x, y) {
this._.push("M", this._x0 = this._x1 = +x, ",", this._y0 = this._y1 = +y);
},
closePath: function() {
if (this._x1 !== null) {
this._x1 = this._x0, this._y1 = this._y0;
this._.push("Z");
}
},
lineTo: function(x, y) {
this._.push("L", this._x1 = +x, ",", this._y1 = +y);
},
quadraticCurveTo: function(x1, y1, x, y) {
this._.push("Q", +x1, ",", +y1, ",", this._x1 = +x, ",", this._y1 = +y);
},
bezierCurveTo: function(x1, y1, x2, y2, x, y) {
this._.push("C", +x1, ",", +y1, ",", +x2, ",", +y2, ",", this._x1 = +x, ",", this._y1 = +y);
},
arcTo: function(x1, y1, x2, y2, r) {
x1 = +x1, y1 = +y1, x2 = +x2, y2 = +y2, r = +r;
var x0 = this._x1,
y0 = this._y1,
x21 = x2 - x1,
y21 = y2 - y1,
x01 = x0 - x1,
y01 = y0 - y1,
l01_2 = x01 * x01 + y01 * y01;
// Is the radius negative? Error.
if (r < 0) throw new Error("negative radius: " + r);
// Is this path empty? Move to (x1,y1).
if (this._x1 === null) {
this._.push(
"M", this._x1 = x1, ",", this._y1 = y1
);
}
// Or, is (x1,y1) coincident with (x0,y0)? Do nothing.
else if (!(l01_2 > epsilon));
// Or, are (x0,y0), (x1,y1) and (x2,y2) collinear?
// Equivalently, is (x1,y1) coincident with (x2,y2)?
// Or, is the radius zero? Line to (x1,y1).
else if (!(Math.abs(y01 * x21 - y21 * x01) > epsilon) || !r) {
this._.push(
"L", this._x1 = x1, ",", this._y1 = y1
);
}
// Otherwise, draw an arc!
else {
var x20 = x2 - x0,
y20 = y2 - y0,
l21_2 = x21 * x21 + y21 * y21,
l20_2 = x20 * x20 + y20 * y20,
l21 = Math.sqrt(l21_2),
l01 = Math.sqrt(l01_2),
l = r * Math.tan((pi$1 - Math.acos((l21_2 + l01_2 - l20_2) / (2 * l21 * l01))) / 2),
t01 = l / l01,
t21 = l / l21;
// If the start tangent is not coincident with (x0,y0), line to.
if (Math.abs(t01 - 1) > epsilon) {
this._.push(
"L", x1 + t01 * x01, ",", y1 + t01 * y01
);
}
this._.push(
"A", r, ",", r, ",0,0,", +(y01 * x20 > x01 * y20), ",", this._x1 = x1 + t21 * x21, ",", this._y1 = y1 + t21 * y21
);
}
},
arc: function(x, y, r, a0, a1, ccw) {
x = +x, y = +y, r = +r;
var dx = r * Math.cos(a0),
dy = r * Math.sin(a0),
x0 = x + dx,
y0 = y + dy,
cw = 1 ^ ccw,
da = ccw ? a0 - a1 : a1 - a0;
// Is the radius negative? Error.
if (r < 0) throw new Error("negative radius: " + r);
// Is this path empty? Move to (x0,y0).
if (this._x1 === null) {
this._.push(
"M", x0, ",", y0
);
}
// Or, is (x0,y0) not coincident with the previous point? Line to (x0,y0).
else if (Math.abs(this._x1 - x0) > epsilon || Math.abs(this._y1 - y0) > epsilon) {
this._.push(
"L", x0, ",", y0
);
}
// Is this arc empty? We’re done.
if (!r) return;
// Is this a complete circle? Draw two arcs to complete the circle.
if (da > tauEpsilon) {
this._.push(
"A", r, ",", r, ",0,1,", cw, ",", x - dx, ",", y - dy,
"A", r, ",", r, ",0,1,", cw, ",", this._x1 = x0, ",", this._y1 = y0
);
}
// Otherwise, draw an arc!
else {
if (da < 0) da = da % tau$1 + tau$1;
this._.push(
"A", r, ",", r, ",0,", +(da >= pi$1), ",", cw, ",", this._x1 = x + r * Math.cos(a1), ",", this._y1 = y + r * Math.sin(a1)
);
}
},
rect: function(x, y, w, h) {
this._.push("M", this._x0 = this._x1 = +x, ",", this._y0 = this._y1 = +y, "h", +w, "v", +h, "h", -w, "Z");
},
toString: function() {
return this._.join("");
}
};
function tree_add(d) {
var x = +this._x.call(null, d),
y = +this._y.call(null, d);
return add(this.cover(x, y), x, y, d);
}
function add(tree, x, y, d) {
if (isNaN(x) || isNaN(y)) return tree; // ignore invalid points
var parent,
node = tree._root,
leaf = {data: d},
x0 = tree._x0,
y0 = tree._y0,
x1 = tree._x1,
y1 = tree._y1,
xm,
ym,
xp,
yp,
right,
bottom,
i,
j;
// If the tree is empty, initialize the root as a leaf.
if (!node) return tree._root = leaf, tree;
// Find the existing leaf for the new point, or add it.
while (node.length) {
if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
if (parent = node, !(node = node[i = bottom << 1 | right])) return parent[i] = leaf, tree;
}
// Is the new point is exactly coincident with the existing point?
xp = +tree._x.call(null, node.data);
yp = +tree._y.call(null, node.data);
if (x === xp && y === yp) return leaf.next = node, parent ? parent[i] = leaf : tree._root = leaf, tree;
// Otherwise, split the leaf node until the old and new point are separated.
do {
parent = parent ? parent[i] = new Array(4) : tree._root = new Array(4);
if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
} while ((i = bottom << 1 | right) === (j = (yp >= ym) << 1 | (xp >= xm)));
return parent[j] = node, parent[i] = leaf, tree;
}
function addAll(data) {
var d, i, n = data.length,
x,
y,
xz = new Array(n),
yz = new Array(n),
x0 = Infinity,
y0 = Infinity,
x1 = -Infinity,
y1 = -Infinity;
// Compute the points and their extent.
for (i = 0; i < n; ++i) {
if (isNaN(x = +this._x.call(null, d = data[i])) || isNaN(y = +this._y.call(null, d))) continue;
xz[i] = x;
yz[i] = y;
if (x < x0) x0 = x;
if (x > x1) x1 = x;
if (y < y0) y0 = y;
if (y > y1) y1 = y;
}
// If there were no (valid) points, inherit the existing extent.
if (x1 < x0) x0 = this._x0, x1 = this._x1;
if (y1 < y0) y0 = this._y0, y1 = this._y1;
// Expand the tree to cover the new points.
this.cover(x0, y0).cover(x1, y1);
// Add the new points.
for (i = 0; i < n; ++i) {
add(this, xz[i], yz[i], data[i]);
}
return this;
}
function tree_cover(x, y) {
if (isNaN(x = +x) || isNaN(y = +y)) return this; // ignore invalid points
var x0 = this._x0,
y0 = this._y0,
x1 = this._x1,
y1 = this._y1;
// If the quadtree has no extent, initialize them.
// Integer extent are necessary so that if we later double the extent,
// the existing quadrant boundaries don’t change due to floating point error!
if (isNaN(x0)) {
x1 = (x0 = Math.floor(x)) + 1;
y1 = (y0 = Math.floor(y)) + 1;
}
// Otherwise, double repeatedly to cover.
else if (x0 > x || x > x1 || y0 > y || y > y1) {
var z = x1 - x0,
node = this._root,
parent,
i;
switch (i = (y < (y0 + y1) / 2) << 1 | (x < (x0 + x1) / 2)) {
case 0: {
do parent = new Array(4), parent[i] = node, node = parent;
while (z *= 2, x1 = x0 + z, y1 = y0 + z, x > x1 || y > y1);
break;
}
case 1: {
do parent = new Array(4), parent[i] = node, node = parent;
while (z *= 2, x0 = x1 - z, y1 = y0 + z, x0 > x || y > y1);
break;
}
case 2: {
do parent = new Array(4), parent[i] = node, node = parent;
while (z *= 2, x1 = x0 + z, y0 = y1 - z, x > x1 || y0 > y);
break;
}
case 3: {
do parent = new Array(4), parent[i] = node, node = parent;
while (z *= 2, x0 = x1 - z, y0 = y1 - z, x0 > x || y0 > y);
break;
}
}
if (this._root && this._root.length) this._root = node;
}
// If the quadtree covers the point already, just return.
else return this;
this._x0 = x0;
this._y0 = y0;
this._x1 = x1;
this._y1 = y1;
return this;
}
function tree_data() {
var data = [];
this.visit(function(node) {
if (!node.length) do data.push(node.data); while (node = node.next)
});
return data;
}
function tree_extent(_) {
return arguments.length
? this.cover(+_[0][0], +_[0][1]).cover(+_[1][0], +_[1][1])
: isNaN(this._x0) ? undefined : [[this._x0, this._y0], [this._x1, this._y1]];
}
function Quad(node, x0, y0, x1, y1) {
this.node = node;
this.x0 = x0;
this.y0 = y0;
this.x1 = x1;
this.y1 = y1;
}
function tree_find(x, y, radius) {
var data,
x0 = this._x0,
y0 = this._y0,
x1,
y1,
x2,
y2,
x3 = this._x1,
y3 = this._y1,
quads = [],
node = this._root,
q,
i;
if (node) quads.push(new Quad(node, x0, y0, x3, y3));
if (radius == null) radius = Infinity;
else {
x0 = x - radius, y0 = y - radius;
x3 = x + radius, y3 = y + radius;
radius *= radius;
}
while (q = quads.pop()) {
// Stop searching if this quadrant can’t contain a closer node.
if (!(node = q.node)
|| (x1 = q.x0) > x3
|| (y1 = q.y0) > y3
|| (x2 = q.x1) < x0
|| (y2 = q.y1) < y0) continue;
// Bisect the current quadrant.
if (node.length) {
var xm = (x1 + x2) / 2,
ym = (y1 + y2) / 2;
quads.push(
new Quad(node[3], xm, ym, x2, y2),
new Quad(node[2], x1, ym, xm, y2),
new Quad(node[1], xm, y1, x2, ym),
new Quad(node[0], x1, y1, xm, ym)
);
// Visit the closest quadrant first.
if (i = (y >= ym) << 1 | (x >= xm)) {
q = quads[quads.length - 1];
quads[quads.length - 1] = quads[quads.length - 1 - i];
quads[quads.length - 1 - i] = q;
}
}
// Visit this point. (Visiting coincident points isn’t necessary!)
else {
var dx = x - +this._x.call(null, node.data),
dy = y - +this._y.call(null, node.data),
d2 = dx * dx + dy * dy;
if (d2 < radius) {
var d = Math.sqrt(radius = d2);
x0 = x - d, y0 = y - d;
x3 = x + d, y3 = y + d;
data = node.data;
}
}
}
return data;
}
function tree_remove(d) {
if (isNaN(x = +this._x.call(null, d)) || isNaN(y = +this._y.call(null, d))) return this; // ignore invalid points
var parent,
node = this._root,
retainer,
previous,
next,
x0 = this._x0,
y0 = this._y0,
x1 = this._x1,
y1 = this._y1,
x,
y,
xm,
ym,
right,
bottom,
i,
j;
// If the tree is empty, initialize the root as a leaf.
if (!node) return this;
// Find the leaf node for the point.
// While descending, also retain the deepest parent with a non-removed sibling.
if (node.length) while (true) {
if (right = x >= (xm = (x0 + x1) / 2)) x0 = xm; else x1 = xm;
if (bottom = y >= (ym = (y0 + y1) / 2)) y0 = ym; else y1 = ym;
if (!(parent = node, node = node[i = bottom << 1 | right])) return this;
if (!node.length) break;
if (parent[(i + 1) & 3] || parent[(i + 2) & 3] || parent[(i + 3) & 3]) retainer = parent, j = i;
}
// Find the point to remove.
while (node.data !== d) if (!(previous = node, node = node.next)) return this;
if (next = node.next) delete node.next;
// If there are multiple coincident points, remove just the point.
if (previous) return (next ? previous.next = next : delete previous.next), this;
// If this is the root point, remove it.
if (!parent) return this._root = next, this;
// Remove this leaf.
next ? parent[i] = next : delete parent[i];
// If the parent now contains exactly one leaf, collapse superfluous parents.
if ((node = parent[0] || parent[1] || parent[2] || parent[3])
&& node === (parent[3] || parent[2] || parent[1] || parent[0])
&& !node.length) {
if (retainer) retainer[j] = node;
else this._root = node;
}
return this;
}
function removeAll(data) {
for (var i = 0, n = data.length; i < n; ++i) this.remove(data[i]);
return this;
}
function tree_root() {
return this._root;
}
function tree_size() {
var size = 0;
this.visit(function(node) {
if (!node.length) do ++size; while (node = node.next)
});
return size;
}
function tree_visit(callback) {
var quads = [], q, node = this._root, child, x0, y0, x1, y1;
if (node) quads.push(new Quad(node, this._x0, this._y0, this._x1, this._y1));
while (q = quads.pop()) {
if (!callback(node = q.node, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1) && node.length) {
var xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
if (child = node[3]) quads.push(new Quad(child, xm, ym, x1, y1));
if (child = node[2]) quads.push(new Quad(child, x0, ym, xm, y1));
if (child = node[1]) quads.push(new Quad(child, xm, y0, x1, ym));
if (child = node[0]) quads.push(new Quad(child, x0, y0, xm, ym));
}
}
return this;
}
function tree_visitAfter(callback) {
var quads = [], next = [], q;
if (this._root) quads.push(new Quad(this._root, this._x0, this._y0, this._x1, this._y1));
while (q = quads.pop()) {
var node = q.node;
if (node.length) {
var child, x0 = q.x0, y0 = q.y0, x1 = q.x1, y1 = q.y1, xm = (x0 + x1) / 2, ym = (y0 + y1) / 2;
if (child = node[0]) quads.push(new Quad(child, x0, y0, xm, ym));
if (child = node[1]) quads.push(new Quad(child, xm, y0, x1, ym));
if (child = node[2]) quads.push(new Quad(child, x0, ym, xm, y1));
if (child = node[3]) quads.push(new Quad(child, xm, ym, x1, y1));
}
next.push(q);
}
while (q = next.pop()) {
callback(q.node, q.x0, q.y0, q.x1, q.y1);
}
return this;
}
function defaultX(d) {
return d[0];
}
function tree_x(_) {
return arguments.length ? (this._x = _, this) : this._x;
}
function defaultY(d) {
return d[1];
}
function tree_y(_) {
return arguments.length ? (this._y = _, this) : this._y;
}
function quadtree(nodes, x, y) {
var tree = new Quadtree(x == null ? defaultX : x, y == null ? defaultY : y, NaN, NaN, NaN, NaN);
return nodes == null ? tree : tree.addAll(nodes);
}
function Quadtree(x, y, x0, y0, x1, y1) {
this._x = x;
this._y = y;
this._x0 = x0;
this._y0 = y0;
this._x1 = x1;
this._y1 = y1;
this._root = undefined;
}
function leaf_copy(leaf) {
var copy = {data: leaf.data}, next = copy;
while (leaf = leaf.next) next = next.next = {data: leaf.data};
return copy;
}
var treeProto = quadtree.prototype = Quadtree.prototype;
treeProto.copy = function() {
var copy = new Quadtree(this._x, this._y, this._x0, this._y0, this._x1, this._y1),
node = this._root,
nodes,
child;
if (!node) return copy;
if (!node.length) return copy._root = leaf_copy(node), copy;
nodes = [{source: node, target: copy._root = new Array(4)}];
while (node = nodes.pop()) {
for (var i = 0; i < 4; ++i) {
if (child = node.source[i]) {
if (child.length) nodes.push({source: child, target: node.target[i] = new Array(4)});
else node.target[i] = leaf_copy(child);
}
}
}
return copy;
};
treeProto.add = tree_add;
treeProto.addAll = addAll;
treeProto.cover = tree_cover;
treeProto.data = tree_data;
treeProto.extent = tree_extent;
treeProto.find = tree_find;
treeProto.remove = tree_remove;
treeProto.removeAll = removeAll;
treeProto.root = tree_root;
treeProto.size = tree_size;
treeProto.visit = tree_visit;
treeProto.visitAfter = tree_visitAfter;
treeProto.x = tree_x;
treeProto.y = tree_y;
var slice$1 = [].slice;
var noabort = {};
function Queue(size) {
if (!(size >= 1)) throw new Error;
this._size = size;
this._call =
this._error = null;
this._tasks = [];
this._data = [];
this._waiting =
this._active =
this._ended =
this._start = 0; // inside a synchronous task callback?
}
Queue.prototype = queue.prototype = {
constructor: Queue,
defer: function(callback) {
if (typeof callback !== "function" || this._call) throw new Error;
if (this._error != null) return this;
var t = slice$1.call(arguments, 1);
t.push(callback);
++this._waiting, this._tasks.push(t);
poke(this);
return this;
},
abort: function() {
if (this._error == null) abort(this, new Error("abort"));
return this;
},
await: function(callback) {
if (typeof callback !== "function" || this._call) throw new Error;
this._call = function(error, results) { callback.apply(null, [error].concat(results)); };
maybeNotify(this);
return this;
},
awaitAll: function(callback) {
if (typeof callback !== "function" || this._call) throw new Error;
this._call = callback;
maybeNotify(this);
return this;
}
};
function poke(q) {
if (!q._start) {
try { start(q); } // let the current task complete
catch (e) {
if (q._tasks[q._ended + q._active - 1]) abort(q, e); // task errored synchronously
else if (!q._data) throw e; // await callback errored synchronously
}
}
}
function start(q) {
while (q._start = q._waiting && q._active < q._size) {
var i = q._ended + q._active,
t = q._tasks[i],
j = t.length - 1,
c = t[j];
t[j] = end(q, i);
--q._waiting, ++q._active;
t = c.apply(null, t);
if (!q._tasks[i]) continue; // task finished synchronously
q._tasks[i] = t || noabort;
}
}
function end(q, i) {
return function(e, r) {
if (!q._tasks[i]) return; // ignore multiple callbacks
--q._active, ++q._ended;
q._tasks[i] = null;
if (q._error != null) return; // ignore secondary errors
if (e != null) {
abort(q, e);
} else {
q._data[i] = r;
if (q._waiting) poke(q);
else maybeNotify(q);
}
};
}
function abort(q, e) {
var i = q._tasks.length, t;
q._error = e; // ignore active callbacks
q._data = undefined; // allow gc
q._waiting = NaN; // prevent starting
while (--i >= 0) {
if (t = q._tasks[i]) {
q._tasks[i] = null;
if (t.abort) {
try { t.abort(); }
catch (e) { /* ignore */ }
}
}
}
q._active = NaN; // allow notification
maybeNotify(q);
}
function maybeNotify(q) {
if (!q._active && q._call) {
var d = q._data;
q._data = undefined; // allow gc
q._call(q._error, d);
}
}
function queue(concurrency) {
return new Queue(arguments.length ? +concurrency : Infinity);
}
function constant$1(x) {
return function constant() {
return x;
};
}
var epsilon$1 = 1e-12;
var pi$2 = Math.PI;
var halfPi$1 = pi$2 / 2;
var tau$2 = 2 * pi$2;
function arcInnerRadius(d) {
return d.innerRadius;
}
function arcOuterRadius(d) {
return d.outerRadius;
}
function arcStartAngle(d) {
return d.startAngle;
}
function arcEndAngle(d) {
return d.endAngle;
}
function arcPadAngle(d) {
return d && d.padAngle; // Note: optional!
}
function asin(x) {
return x >= 1 ? halfPi$1 : x <= -1 ? -halfPi$1 : Math.asin(x);
}
function intersect(x0, y0, x1, y1, x2, y2, x3, y3) {
var x10 = x1 - x0, y10 = y1 - y0,
x32 = x3 - x2, y32 = y3 - y2,
t = (x32 * (y0 - y2) - y32 * (x0 - x2)) / (y32 * x10 - x32 * y10);
return [x0 + t * x10, y0 + t * y10];
}
// Compute perpendicular offset line of length rc.
// http://mathworld.wolfram.com/Circle-LineIntersection.html
function cornerTangents(x0, y0, x1, y1, r1, rc, cw) {
var x01 = x0 - x1,
y01 = y0 - y1,
lo = (cw ? rc : -rc) / Math.sqrt(x01 * x01 + y01 * y01),
ox = lo * y01,
oy = -lo * x01,
x11 = x0 + ox,
y11 = y0 + oy,
x10 = x1 + ox,
y10 = y1 + oy,
x00 = (x11 + x10) / 2,
y00 = (y11 + y10) / 2,
dx = x10 - x11,
dy = y10 - y11,
d2 = dx * dx + dy * dy,
r = r1 - rc,
D = x11 * y10 - x10 * y11,
d = (dy < 0 ? -1 : 1) * Math.sqrt(Math.max(0, r * r * d2 - D * D)),
cx0 = (D * dy - dx * d) / d2,
cy0 = (-D * dx - dy * d) / d2,
cx1 = (D * dy + dx * d) / d2,
cy1 = (-D * dx + dy * d) / d2,
dx0 = cx0 - x00,
dy0 = cy0 - y00,
dx1 = cx1 - x00,
dy1 = cy1 - y00;
// Pick the closer of the two intersection points.
// TODO Is there a faster way to determine which intersection to use?
if (dx0 * dx0 + dy0 * dy0 > dx1 * dx1 + dy1 * dy1) cx0 = cx1, cy0 = cy1;
return {
cx: cx0,
cy: cy0,
x01: -ox,
y01: -oy,
x11: cx0 * (r1 / r - 1),
y11: cy0 * (r1 / r - 1)
};
}
function arc() {
var innerRadius = arcInnerRadius,
outerRadius = arcOuterRadius,
cornerRadius = constant$1(0),
padRadius = null,
startAngle = arcStartAngle,
endAngle = arcEndAngle,
padAngle = arcPadAngle,
context = null;
function arc() {
var buffer,
r,
r0 = +innerRadius.apply(this, arguments),
r1 = +outerRadius.apply(this, arguments),
a0 = startAngle.apply(this, arguments) - halfPi$1,
a1 = endAngle.apply(this, arguments) - halfPi$1,
da = Math.abs(a1 - a0),
cw = a1 > a0;
if (!context) context = buffer = path();
// Ensure that the outer radius is always larger than the inner radius.
if (r1 < r0) r = r1, r1 = r0, r0 = r;
// Is it a point?
if (!(r1 > epsilon$1)) context.moveTo(0, 0);
// Or is it a circle or annulus?
else if (da > tau$2 - epsilon$1) {
context.moveTo(r1 * Math.cos(a0), r1 * Math.sin(a0));
context.arc(0, 0, r1, a0, a1, !cw);
if (r0 > epsilon$1) {
context.moveTo(r0 * Math.cos(a1), r0 * Math.sin(a1));
context.arc(0, 0, r0, a1, a0, cw);
}
}
// Or is it a circular or annular sector?
else {
var a01 = a0,
a11 = a1,
a00 = a0,
a10 = a1,
da0 = da,
da1 = da,
ap = padAngle.apply(this, arguments) / 2,
rp = (ap > epsilon$1) && (padRadius ? +padRadius.apply(this, arguments) : Math.sqrt(r0 * r0 + r1 * r1)),
rc = Math.min(Math.abs(r1 - r0) / 2, +cornerRadius.apply(this, arguments)),
rc0 = rc,
rc1 = rc,
t0,
t1;
// Apply padding? Note that since r1 ≥ r0, da1 ≥ da0.
if (rp > epsilon$1) {
var p0 = asin(rp / r0 * Math.sin(ap)),
p1 = asin(rp / r1 * Math.sin(ap));
if ((da0 -= p0 * 2) > epsilon$1) p0 *= (cw ? 1 : -1), a00 += p0, a10 -= p0;
else da0 = 0, a00 = a10 = (a0 + a1) / 2;
if ((da1 -= p1 * 2) > epsilon$1) p1 *= (cw ? 1 : -1), a01 += p1, a11 -= p1;
else da1 = 0, a01 = a11 = (a0 + a1) / 2;
}
var x01 = r1 * Math.cos(a01),
y01 = r1 * Math.sin(a01),
x10 = r0 * Math.cos(a10),
y10 = r0 * Math.sin(a10);
// Apply rounded corners?
if (rc > epsilon$1) {
var x11 = r1 * Math.cos(a11),
y11 = r1 * Math.sin(a11),
x00 = r0 * Math.cos(a00),
y00 = r0 * Math.sin(a00);
// Restrict the corner radius according to the sector angle.
if (da < pi$2) {
var oc = da0 > epsilon$1 ? intersect(x01, y01, x00, y00, x11, y11, x10, y10) : [x10, y10],
ax = x01 - oc[0],
ay = y01 - oc[1],
bx = x11 - oc[0],
by = y11 - oc[1],
kc = 1 / Math.sin(Math.acos((ax * bx + ay * by) / (Math.sqrt(ax * ax + ay * ay) * Math.sqrt(bx * bx + by * by))) / 2),
lc = Math.sqrt(oc[0] * oc[0] + oc[1] * oc[1]);
rc0 = Math.min(rc, (r0 - lc) / (kc - 1));
rc1 = Math.min(rc, (r1 - lc) / (kc + 1));
}
}
// Is the sector collapsed to a line?
if (!(da1 > epsilon$1)) context.moveTo(x01, y01);
// Does the sector’s outer ring have rounded corners?
else if (rc1 > epsilon$1) {
t0 = cornerTangents(x00, y00, x01, y01, r1, rc1, cw);
t1 = cornerTangents(x11, y11, x10, y10, r1, rc1, cw);
context.moveTo(t0.cx + t0.x01, t0.cy + t0.y01);
// Have the corners merged?
if (rc1 < rc) context.arc(t0.cx, t0.cy, rc1, Math.atan2(t0.y01, t0.x01), Math.atan2(t1.y01, t1.x01), !cw);
// Otherwise, draw the two corners and the ring.
else {
context.arc(t0.cx, t0.cy, rc1, Math.atan2(t0.y01, t0.x01), Math.atan2(t0.y11, t0.x11), !cw);
context.arc(0, 0, r1, Math.atan2(t0.cy + t0.y11, t0.cx + t0.x11), Math.atan2(t1.cy + t1.y11, t1.cx + t1.x11), !cw);
context.arc(t1.cx, t1.cy, rc1, Math.atan2(t1.y11, t1.x11), Math.atan2(t1.y01, t1.x01), !cw);
}
}
// Or is the outer ring just a circular arc?
else context.moveTo(x01, y01), context.arc(0, 0, r1, a01, a11, !cw);
// Is there no inner ring, and it’s a circular sector?
// Or perhaps it’s an annular sector collapsed due to padding?
if (!(r0 > epsilon$1) || !(da0 > epsilon$1)) context.lineTo(x10, y10);
// Does the sector’s inner ring (or point) have rounded corners?
else if (rc0 > epsilon$1) {
t0 = cornerTangents(x10, y10, x11, y11, r0, -rc0, cw);
t1 = cornerTangents(x01, y01, x00, y00, r0, -rc0, cw);
context.lineTo(t0.cx + t0.x01, t0.cy + t0.y01);
// Have the corners merged?
if (rc0 < rc) context.arc(t0.cx, t0.cy, rc0, Math.atan2(t0.y01, t0.x01), Math.atan2(t1.y01, t1.x01), !cw);
// Otherwise, draw the two corners and the ring.
else {
context.arc(t0.cx, t0.cy, rc0, Math.atan2(t0.y01, t0.x01), Math.atan2(t0.y11, t0.x11), !cw);
context.arc(0, 0, r0, Math.atan2(t0.cy + t0.y11, t0.cx + t0.x11), Math.atan2(t1.cy + t1.y11, t1.cx + t1.x11), cw);
context.arc(t1.cx, t1.cy, rc0, Math.atan2(t1.y11, t1.x11), Math.atan2(t1.y01, t1.x01), !cw);
}
}
// Or is the inner ring just a circular arc?
else context.arc(0, 0, r0, a10, a00, cw);
}
context.closePath();
if (buffer) return context = null, buffer + "" || null;
}
arc.centroid = function() {
var r = (+innerRadius.apply(this, arguments) + +outerRadius.apply(this, arguments)) / 2,
a = (+startAngle.apply(this, arguments) + +endAngle.apply(this, arguments)) / 2 - pi$2 / 2;
return [Math.cos(a) * r, Math.sin(a) * r];
};
arc.innerRadius = function(_) {
return arguments.length ? (innerRadius = typeof _ === "function" ? _ : constant$1(+_), arc) : innerRadius;
};
arc.outerRadius = function(_) {
return arguments.length ? (outerRadius = typeof _ === "function" ? _ : constant$1(+_), arc) : outerRadius;
};
arc.cornerRadius = function(_) {
return arguments.length ? (cornerRadius = typeof _ === "function" ? _ : constant$1(+_), arc) : cornerRadius;
};
arc.padRadius = function(_) {
return arguments.length ? (padRadius = _ == null ? null : typeof _ === "function" ? _ : constant$1(+_), arc) : padRadius;
};
arc.startAngle = function(_) {
return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$1(+_), arc) : startAngle;
};
arc.endAngle = function(_) {
return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$1(+_), arc) : endAngle;
};
arc.padAngle = function(_) {
return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$1(+_), arc) : padAngle;
};
arc.context = function(_) {
return arguments.length ? ((context = _ == null ? null : _), arc) : context;
};
return arc;
}
function Linear(context) {
this._context = context;
}
Linear.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._point = 0;
},
lineEnd: function() {
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; // proceed
default: this._context.lineTo(x, y); break;
}
}
};
function curveLinear(context) {
return new Linear(context);
}
function x(p) {
return p[0];
}
function y(p) {
return p[1];
}
function line() {
var x$$ = x,
y$$ = y,
defined = constant$1(true),
context = null,
curve = curveLinear,
output = null;
function line(data) {
var i,
n = data.length,
d,
defined0 = false,
buffer;
if (context == null) output = curve(buffer = path());
for (i = 0; i <= n; ++i) {
if (!(i < n && defined(d = data[i], i, data)) === defined0) {
if (defined0 = !defined0) output.lineStart();
else output.lineEnd();
}
if (defined0) output.point(+x$$(d, i, data), +y$$(d, i, data));
}
if (buffer) return output = null, buffer + "" || null;
}
line.x = function(_) {
return arguments.length ? (x$$ = typeof _ === "function" ? _ : constant$1(+_), line) : x$$;
};
line.y = function(_) {
return arguments.length ? (y$$ = typeof _ === "function" ? _ : constant$1(+_), line) : y$$;
};
line.defined = function(_) {
return arguments.length ? (defined = typeof _ === "function" ? _ : constant$1(!!_), line) : defined;
};
line.curve = function(_) {
return arguments.length ? (curve = _, context != null && (output = curve(context)), line) : curve;
};
line.context = function(_) {
return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), line) : context;
};
return line;
}
function area$1() {
var x0 = x,
x1 = null,
y0 = constant$1(0),
y1 = y,
defined = constant$1(true),
context = null,
curve = curveLinear,
output = null;
function area(data) {
var i,
j,
k,
n = data.length,
d,
defined0 = false,
buffer,
x0z = new Array(n),
y0z = new Array(n);
if (context == null) output = curve(buffer = path());
for (i = 0; i <= n; ++i) {
if (!(i < n && defined(d = data[i], i, data)) === defined0) {
if (defined0 = !defined0) {
j = i;
output.areaStart();
output.lineStart();
} else {
output.lineEnd();
output.lineStart();
for (k = i - 1; k >= j; --k) {
output.point(x0z[k], y0z[k]);
}
output.lineEnd();
output.areaEnd();
}
}
if (defined0) {
x0z[i] = +x0(d, i, data), y0z[i] = +y0(d, i, data);
output.point(x1 ? +x1(d, i, data) : x0z[i], y1 ? +y1(d, i, data) : y0z[i]);
}
}
if (buffer) return output = null, buffer + "" || null;
}
function arealine() {
return line().defined(defined).curve(curve).context(context);
}
area.x = function(_) {
return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$1(+_), x1 = null, area) : x0;
};
area.x0 = function(_) {
return arguments.length ? (x0 = typeof _ === "function" ? _ : constant$1(+_), area) : x0;
};
area.x1 = function(_) {
return arguments.length ? (x1 = _ == null ? null : typeof _ === "function" ? _ : constant$1(+_), area) : x1;
};
area.y = function(_) {
return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$1(+_), y1 = null, area) : y0;
};
area.y0 = function(_) {
return arguments.length ? (y0 = typeof _ === "function" ? _ : constant$1(+_), area) : y0;
};
area.y1 = function(_) {
return arguments.length ? (y1 = _ == null ? null : typeof _ === "function" ? _ : constant$1(+_), area) : y1;
};
area.lineX0 =
area.lineY0 = function() {
return arealine().x(x0).y(y0);
};
area.lineY1 = function() {
return arealine().x(x0).y(y1);
};
area.lineX1 = function() {
return arealine().x(x1).y(y0);
};
area.defined = function(_) {
return arguments.length ? (defined = typeof _ === "function" ? _ : constant$1(!!_), area) : defined;
};
area.curve = function(_) {
return arguments.length ? (curve = _, context != null && (output = curve(context)), area) : curve;
};
area.context = function(_) {
return arguments.length ? (_ == null ? context = output = null : output = curve(context = _), area) : context;
};
return area;
}
function descending$1(a, b) {
return b < a ? -1 : b > a ? 1 : b >= a ? 0 : NaN;
}
function identity$1(d) {
return d;
}
function pie() {
var value = identity$1,
sortValues = descending$1,
sort = null,
startAngle = constant$1(0),
endAngle = constant$1(tau$2),
padAngle = constant$1(0);
function pie(data) {
var i,
n = data.length,
j,
k,
sum = 0,
index = new Array(n),
arcs = new Array(n),
a0 = +startAngle.apply(this, arguments),
da = Math.min(tau$2, Math.max(-tau$2, endAngle.apply(this, arguments) - a0)),
a1,
p = Math.min(Math.abs(da) / n, padAngle.apply(this, arguments)),
pa = p * (da < 0 ? -1 : 1),
v;
for (i = 0; i < n; ++i) {
if ((v = arcs[index[i] = i] = +value(data[i], i, data)) > 0) {
sum += v;
}
}
// Optionally sort the arcs by previously-computed values or by data.
if (sortValues != null) index.sort(function(i, j) { return sortValues(arcs[i], arcs[j]); });
else if (sort != null) index.sort(function(i, j) { return sort(data[i], data[j]); });
// Compute the arcs! They are stored in the original data's order.
for (i = 0, k = sum ? (da - n * pa) / sum : 0; i < n; ++i, a0 = a1) {
j = index[i], v = arcs[j], a1 = a0 + (v > 0 ? v * k : 0) + pa, arcs[j] = {
data: data[j],
index: i,
value: v,
startAngle: a0,
endAngle: a1,
padAngle: p
};
}
return arcs;
}
pie.value = function(_) {
return arguments.length ? (value = typeof _ === "function" ? _ : constant$1(+_), pie) : value;
};
pie.sortValues = function(_) {
return arguments.length ? (sortValues = _, sort = null, pie) : sortValues;
};
pie.sort = function(_) {
return arguments.length ? (sort = _, sortValues = null, pie) : sort;
};
pie.startAngle = function(_) {
return arguments.length ? (startAngle = typeof _ === "function" ? _ : constant$1(+_), pie) : startAngle;
};
pie.endAngle = function(_) {
return arguments.length ? (endAngle = typeof _ === "function" ? _ : constant$1(+_), pie) : endAngle;
};
pie.padAngle = function(_) {
return arguments.length ? (padAngle = typeof _ === "function" ? _ : constant$1(+_), pie) : padAngle;
};
return pie;
}
var curveRadialLinear = curveRadial(curveLinear);
function Radial(curve) {
this._curve = curve;
}
Radial.prototype = {
areaStart: function() {
this._curve.areaStart();
},
areaEnd: function() {
this._curve.areaEnd();
},
lineStart: function() {
this._curve.lineStart();
},
lineEnd: function() {
this._curve.lineEnd();
},
point: function(a, r) {
this._curve.point(r * Math.sin(a), r * -Math.cos(a));
}
};
function curveRadial(curve) {
function radial(context) {
return new Radial(curve(context));
}
radial._curve = curve;
return radial;
}
function radialLine(l) {
var c = l.curve;
l.angle = l.x, delete l.x;
l.radius = l.y, delete l.y;
l.curve = function(_) {
return arguments.length ? c(curveRadial(_)) : c()._curve;
};
return l;
}
function radialLine$1() {
return radialLine(line().curve(curveRadialLinear));
}
function radialArea() {
var a = area$1().curve(curveRadialLinear),
c = a.curve,
x0 = a.lineX0,
x1 = a.lineX1,
y0 = a.lineY0,
y1 = a.lineY1;
a.angle = a.x, delete a.x;
a.startAngle = a.x0, delete a.x0;
a.endAngle = a.x1, delete a.x1;
a.radius = a.y, delete a.y;
a.innerRadius = a.y0, delete a.y0;
a.outerRadius = a.y1, delete a.y1;
a.lineStartAngle = function() { return radialLine(x0()); }, delete a.lineX0;
a.lineEndAngle = function() { return radialLine(x1()); }, delete a.lineX1;
a.lineInnerRadius = function() { return radialLine(y0()); }, delete a.lineY0;
a.lineOuterRadius = function() { return radialLine(y1()); }, delete a.lineY1;
a.curve = function(_) {
return arguments.length ? c(curveRadial(_)) : c()._curve;
};
return a;
}
var circle = {
draw: function(context, size) {
var r = Math.sqrt(size / pi$2);
context.moveTo(r, 0);
context.arc(0, 0, r, 0, tau$2);
}
};
var cross$1 = {
draw: function(context, size) {
var r = Math.sqrt(size / 5) / 2;
context.moveTo(-3 * r, -r);
context.lineTo(-r, -r);
context.lineTo(-r, -3 * r);
context.lineTo(r, -3 * r);
context.lineTo(r, -r);
context.lineTo(3 * r, -r);
context.lineTo(3 * r, r);
context.lineTo(r, r);
context.lineTo(r, 3 * r);
context.lineTo(-r, 3 * r);
context.lineTo(-r, r);
context.lineTo(-3 * r, r);
context.closePath();
}
};
var tan30 = Math.sqrt(1 / 3);
var tan30_2 = tan30 * 2;
var diamond = {
draw: function(context, size) {
var y = Math.sqrt(size / tan30_2),
x = y * tan30;
context.moveTo(0, -y);
context.lineTo(x, 0);
context.lineTo(0, y);
context.lineTo(-x, 0);
context.closePath();
}
};
var ka = 0.89081309152928522810;
var kr = Math.sin(pi$2 / 10) / Math.sin(7 * pi$2 / 10);
var kx = Math.sin(tau$2 / 10) * kr;
var ky = -Math.cos(tau$2 / 10) * kr;
var star = {
draw: function(context, size) {
var r = Math.sqrt(size * ka),
x = kx * r,
y = ky * r;
context.moveTo(0, -r);
context.lineTo(x, y);
for (var i = 1; i < 5; ++i) {
var a = tau$2 * i / 5,
c = Math.cos(a),
s = Math.sin(a);
context.lineTo(s * r, -c * r);
context.lineTo(c * x - s * y, s * x + c * y);
}
context.closePath();
}
};
var square = {
draw: function(context, size) {
var w = Math.sqrt(size),
x = -w / 2;
context.rect(x, x, w, w);
}
};
var sqrt3 = Math.sqrt(3);
var triangle = {
draw: function(context, size) {
var y = -Math.sqrt(size / (sqrt3 * 3));
context.moveTo(0, y * 2);
context.lineTo(-sqrt3 * y, -y);
context.lineTo(sqrt3 * y, -y);
context.closePath();
}
};
var c = -0.5;
var s = Math.sqrt(3) / 2;
var k = 1 / Math.sqrt(12);
var a = (k / 2 + 1) * 3;
var wye = {
draw: function(context, size) {
var r = Math.sqrt(size / a),
x0 = r / 2,
y0 = r * k,
x1 = x0,
y1 = r * k + r,
x2 = -x1,
y2 = y1;
context.moveTo(x0, y0);
context.lineTo(x1, y1);
context.lineTo(x2, y2);
context.lineTo(c * x0 - s * y0, s * x0 + c * y0);
context.lineTo(c * x1 - s * y1, s * x1 + c * y1);
context.lineTo(c * x2 - s * y2, s * x2 + c * y2);
context.lineTo(c * x0 + s * y0, c * y0 - s * x0);
context.lineTo(c * x1 + s * y1, c * y1 - s * x1);
context.lineTo(c * x2 + s * y2, c * y2 - s * x2);
context.closePath();
}
};
var symbols = [
circle,
cross$1,
diamond,
square,
star,
triangle,
wye
];
function symbol() {
var type = constant$1(circle),
size = constant$1(64),
context = null;
function symbol() {
var buffer;
if (!context) context = buffer = path();
type.apply(this, arguments).draw(context, +size.apply(this, arguments));
if (buffer) return context = null, buffer + "" || null;
}
symbol.type = function(_) {
return arguments.length ? (type = typeof _ === "function" ? _ : constant$1(_), symbol) : type;
};
symbol.size = function(_) {
return arguments.length ? (size = typeof _ === "function" ? _ : constant$1(+_), symbol) : size;
};
symbol.context = function(_) {
return arguments.length ? (context = _ == null ? null : _, symbol) : context;
};
return symbol;
}
function noop() {}
function point(that, x, y) {
that._context.bezierCurveTo(
(2 * that._x0 + that._x1) / 3,
(2 * that._y0 + that._y1) / 3,
(that._x0 + 2 * that._x1) / 3,
(that._y0 + 2 * that._y1) / 3,
(that._x0 + 4 * that._x1 + x) / 6,
(that._y0 + 4 * that._y1 + y) / 6
);
}
function Basis(context) {
this._context = context;
}
Basis.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 =
this._y0 = this._y1 = NaN;
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 3: point(this, this._x1, this._y1); // proceed
case 2: this._context.lineTo(this._x1, this._y1); break;
}
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; break;
case 2: this._point = 3; this._context.lineTo((5 * this._x0 + this._x1) / 6, (5 * this._y0 + this._y1) / 6); // proceed
default: point(this, x, y); break;
}
this._x0 = this._x1, this._x1 = x;
this._y0 = this._y1, this._y1 = y;
}
};
function basis(context) {
return new Basis(context);
}
function BasisClosed(context) {
this._context = context;
}
BasisClosed.prototype = {
areaStart: noop,
areaEnd: noop,
lineStart: function() {
this._x0 = this._x1 = this._x2 = this._x3 = this._x4 =
this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = NaN;
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 1: {
this._context.moveTo(this._x2, this._y2);
this._context.closePath();
break;
}
case 2: {
this._context.moveTo((this._x2 + 2 * this._x3) / 3, (this._y2 + 2 * this._y3) / 3);
this._context.lineTo((this._x3 + 2 * this._x2) / 3, (this._y3 + 2 * this._y2) / 3);
this._context.closePath();
break;
}
case 3: {
this.point(this._x2, this._y2);
this.point(this._x3, this._y3);
this.point(this._x4, this._y4);
break;
}
}
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._x2 = x, this._y2 = y; break;
case 1: this._point = 2; this._x3 = x, this._y3 = y; break;
case 2: this._point = 3; this._x4 = x, this._y4 = y; this._context.moveTo((this._x0 + 4 * this._x1 + x) / 6, (this._y0 + 4 * this._y1 + y) / 6); break;
default: point(this, x, y); break;
}
this._x0 = this._x1, this._x1 = x;
this._y0 = this._y1, this._y1 = y;
}
};
function basisClosed(context) {
return new BasisClosed(context);
}
function BasisOpen(context) {
this._context = context;
}
BasisOpen.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 =
this._y0 = this._y1 = NaN;
this._point = 0;
},
lineEnd: function() {
if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; break;
case 1: this._point = 2; break;
case 2: this._point = 3; var x0 = (this._x0 + 4 * this._x1 + x) / 6, y0 = (this._y0 + 4 * this._y1 + y) / 6; this._line ? this._context.lineTo(x0, y0) : this._context.moveTo(x0, y0); break;
case 3: this._point = 4; // proceed
default: point(this, x, y); break;
}
this._x0 = this._x1, this._x1 = x;
this._y0 = this._y1, this._y1 = y;
}
};
function basisOpen(context) {
return new BasisOpen(context);
}
function Bundle(context, beta) {
this._basis = new Basis(context);
this._beta = beta;
}
Bundle.prototype = {
lineStart: function() {
this._x = [];
this._y = [];
this._basis.lineStart();
},
lineEnd: function() {
var x = this._x,
y = this._y,
j = x.length - 1;
if (j > 0) {
var x0 = x[0],
y0 = y[0],
dx = x[j] - x0,
dy = y[j] - y0,
i = -1,
t;
while (++i <= j) {
t = i / j;
this._basis.point(
this._beta * x[i] + (1 - this._beta) * (x0 + t * dx),
this._beta * y[i] + (1 - this._beta) * (y0 + t * dy)
);
}
}
this._x = this._y = null;
this._basis.lineEnd();
},
point: function(x, y) {
this._x.push(+x);
this._y.push(+y);
}
};
var bundle = (function custom(beta) {
function bundle(context) {
return beta === 1 ? new Basis(context) : new Bundle(context, beta);
}
bundle.beta = function(beta) {
return custom(+beta);
};
return bundle;
})(0.85);
function point$1(that, x, y) {
that._context.bezierCurveTo(
that._x1 + that._k * (that._x2 - that._x0),
that._y1 + that._k * (that._y2 - that._y0),
that._x2 + that._k * (that._x1 - x),
that._y2 + that._k * (that._y1 - y),
that._x2,
that._y2
);
}
function Cardinal(context, tension) {
this._context = context;
this._k = (1 - tension) / 6;
}
Cardinal.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 = this._x2 =
this._y0 = this._y1 = this._y2 = NaN;
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 2: this._context.lineTo(this._x2, this._y2); break;
case 3: point$1(this, this._x1, this._y1); break;
}
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; this._x1 = x, this._y1 = y; break;
case 2: this._point = 3; // proceed
default: point$1(this, x, y); break;
}
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var cardinal = (function custom(tension) {
function cardinal(context) {
return new Cardinal(context, tension);
}
cardinal.tension = function(tension) {
return custom(+tension);
};
return cardinal;
})(0);
function CardinalClosed(context, tension) {
this._context = context;
this._k = (1 - tension) / 6;
}
CardinalClosed.prototype = {
areaStart: noop,
areaEnd: noop,
lineStart: function() {
this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 1: {
this._context.moveTo(this._x3, this._y3);
this._context.closePath();
break;
}
case 2: {
this._context.lineTo(this._x3, this._y3);
this._context.closePath();
break;
}
case 3: {
this.point(this._x3, this._y3);
this.point(this._x4, this._y4);
this.point(this._x5, this._y5);
break;
}
}
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
default: point$1(this, x, y); break;
}
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var cardinalClosed = (function custom(tension) {
function cardinal(context) {
return new CardinalClosed(context, tension);
}
cardinal.tension = function(tension) {
return custom(+tension);
};
return cardinal;
})(0);
function CardinalOpen(context, tension) {
this._context = context;
this._k = (1 - tension) / 6;
}
CardinalOpen.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 = this._x2 =
this._y0 = this._y1 = this._y2 = NaN;
this._point = 0;
},
lineEnd: function() {
if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; break;
case 1: this._point = 2; break;
case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
case 3: this._point = 4; // proceed
default: point$1(this, x, y); break;
}
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var cardinalOpen = (function custom(tension) {
function cardinal(context) {
return new CardinalOpen(context, tension);
}
cardinal.tension = function(tension) {
return custom(+tension);
};
return cardinal;
})(0);
function point$2(that, x, y) {
var x1 = that._x1,
y1 = that._y1,
x2 = that._x2,
y2 = that._y2;
if (that._l01_a > epsilon$1) {
var a = 2 * that._l01_2a + 3 * that._l01_a * that._l12_a + that._l12_2a,
n = 3 * that._l01_a * (that._l01_a + that._l12_a);
x1 = (x1 * a - that._x0 * that._l12_2a + that._x2 * that._l01_2a) / n;
y1 = (y1 * a - that._y0 * that._l12_2a + that._y2 * that._l01_2a) / n;
}
if (that._l23_a > epsilon$1) {
var b = 2 * that._l23_2a + 3 * that._l23_a * that._l12_a + that._l12_2a,
m = 3 * that._l23_a * (that._l23_a + that._l12_a);
x2 = (x2 * b + that._x1 * that._l23_2a - x * that._l12_2a) / m;
y2 = (y2 * b + that._y1 * that._l23_2a - y * that._l12_2a) / m;
}
that._context.bezierCurveTo(x1, y1, x2, y2, that._x2, that._y2);
}
function CatmullRom(context, alpha) {
this._context = context;
this._alpha = alpha;
}
CatmullRom.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 = this._x2 =
this._y0 = this._y1 = this._y2 = NaN;
this._l01_a = this._l12_a = this._l23_a =
this._l01_2a = this._l12_2a = this._l23_2a =
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 2: this._context.lineTo(this._x2, this._y2); break;
case 3: this.point(this, this._x2, this._y2); break;
}
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
if (this._point) {
var x23 = this._x2 - x,
y23 = this._y2 - y;
this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
}
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; break;
case 2: this._point = 3; // proceed
default: point$2(this, x, y); break;
}
this._l01_a = this._l12_a, this._l12_a = this._l23_a;
this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var catmullRom = (function custom(alpha) {
function catmullRom(context) {
return alpha ? new CatmullRom(context, alpha) : new Cardinal(context, 0);
}
catmullRom.alpha = function(alpha) {
return custom(+alpha);
};
return catmullRom;
})(0.5);
function CatmullRomClosed(context, alpha) {
this._context = context;
this._alpha = alpha;
}
CatmullRomClosed.prototype = {
areaStart: noop,
areaEnd: noop,
lineStart: function() {
this._x0 = this._x1 = this._x2 = this._x3 = this._x4 = this._x5 =
this._y0 = this._y1 = this._y2 = this._y3 = this._y4 = this._y5 = NaN;
this._l01_a = this._l12_a = this._l23_a =
this._l01_2a = this._l12_2a = this._l23_2a =
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 1: {
this._context.moveTo(this._x3, this._y3);
this._context.closePath();
break;
}
case 2: {
this._context.lineTo(this._x3, this._y3);
this._context.closePath();
break;
}
case 3: {
this.point(this._x3, this._y3);
this.point(this._x4, this._y4);
this.point(this._x5, this._y5);
break;
}
}
},
point: function(x, y) {
x = +x, y = +y;
if (this._point) {
var x23 = this._x2 - x,
y23 = this._y2 - y;
this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
}
switch (this._point) {
case 0: this._point = 1; this._x3 = x, this._y3 = y; break;
case 1: this._point = 2; this._context.moveTo(this._x4 = x, this._y4 = y); break;
case 2: this._point = 3; this._x5 = x, this._y5 = y; break;
default: point$2(this, x, y); break;
}
this._l01_a = this._l12_a, this._l12_a = this._l23_a;
this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var catmullRomClosed = (function custom(alpha) {
function catmullRom(context) {
return alpha ? new CatmullRomClosed(context, alpha) : new CardinalClosed(context, 0);
}
catmullRom.alpha = function(alpha) {
return custom(+alpha);
};
return catmullRom;
})(0.5);
function CatmullRomOpen(context, alpha) {
this._context = context;
this._alpha = alpha;
}
CatmullRomOpen.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 = this._x2 =
this._y0 = this._y1 = this._y2 = NaN;
this._l01_a = this._l12_a = this._l23_a =
this._l01_2a = this._l12_2a = this._l23_2a =
this._point = 0;
},
lineEnd: function() {
if (this._line || (this._line !== 0 && this._point === 3)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
if (this._point) {
var x23 = this._x2 - x,
y23 = this._y2 - y;
this._l23_a = Math.sqrt(this._l23_2a = Math.pow(x23 * x23 + y23 * y23, this._alpha));
}
switch (this._point) {
case 0: this._point = 1; break;
case 1: this._point = 2; break;
case 2: this._point = 3; this._line ? this._context.lineTo(this._x2, this._y2) : this._context.moveTo(this._x2, this._y2); break;
case 3: this._point = 4; // proceed
default: point$2(this, x, y); break;
}
this._l01_a = this._l12_a, this._l12_a = this._l23_a;
this._l01_2a = this._l12_2a, this._l12_2a = this._l23_2a;
this._x0 = this._x1, this._x1 = this._x2, this._x2 = x;
this._y0 = this._y1, this._y1 = this._y2, this._y2 = y;
}
};
var catmullRomOpen = (function custom(alpha) {
function catmullRom(context) {
return alpha ? new CatmullRomOpen(context, alpha) : new CardinalOpen(context, 0);
}
catmullRom.alpha = function(alpha) {
return custom(+alpha);
};
return catmullRom;
})(0.5);
function LinearClosed(context) {
this._context = context;
}
LinearClosed.prototype = {
areaStart: noop,
areaEnd: noop,
lineStart: function() {
this._point = 0;
},
lineEnd: function() {
if (this._point) this._context.closePath();
},
point: function(x, y) {
x = +x, y = +y;
if (this._point) this._context.lineTo(x, y);
else this._point = 1, this._context.moveTo(x, y);
}
};
function linearClosed(context) {
return new LinearClosed(context);
}
function sign(x) {
return x < 0 ? -1 : 1;
}
// Calculate the slopes of the tangents (Hermite-type interpolation) based on
// the following paper: Steffen, M. 1990. A Simple Method for Monotonic
// Interpolation in One Dimension. Astronomy and Astrophysics, Vol. 239, NO.
// NOV(II), P. 443, 1990.
function slope3(that, x2, y2) {
var h0 = that._x1 - that._x0,
h1 = x2 - that._x1,
s0 = (that._y1 - that._y0) / (h0 || h1 < 0 && -0),
s1 = (y2 - that._y1) / (h1 || h0 < 0 && -0),
p = (s0 * h1 + s1 * h0) / (h0 + h1);
return (sign(s0) + sign(s1)) * Math.min(Math.abs(s0), Math.abs(s1), 0.5 * Math.abs(p)) || 0;
}
// Calculate a one-sided slope.
function slope2(that, t) {
var h = that._x1 - that._x0;
return h ? (3 * (that._y1 - that._y0) / h - t) / 2 : t;
}
// According to https://en.wikipedia.org/wiki/Cubic_Hermite_spline#Representations
// "you can express cubic Hermite interpolation in terms of cubic Bézier curves
// with respect to the four values p0, p0 + m0 / 3, p1 - m1 / 3, p1".
function point$3(that, t0, t1) {
var x0 = that._x0,
y0 = that._y0,
x1 = that._x1,
y1 = that._y1,
dx = (x1 - x0) / 3;
that._context.bezierCurveTo(x0 + dx, y0 + dx * t0, x1 - dx, y1 - dx * t1, x1, y1);
}
function MonotoneX(context) {
this._context = context;
}
MonotoneX.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x0 = this._x1 =
this._y0 = this._y1 =
this._t0 = NaN;
this._point = 0;
},
lineEnd: function() {
switch (this._point) {
case 2: this._context.lineTo(this._x1, this._y1); break;
case 3: point$3(this, this._t0, slope2(this, this._t0)); break;
}
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
this._line = 1 - this._line;
},
point: function(x, y) {
var t1 = NaN;
x = +x, y = +y;
if (x === this._x1 && y === this._y1) return; // Ignore coincident points.
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; break;
case 2: this._point = 3; point$3(this, slope2(this, t1 = slope3(this, x, y)), t1); break;
default: point$3(this, this._t0, t1 = slope3(this, x, y)); break;
}
this._x0 = this._x1, this._x1 = x;
this._y0 = this._y1, this._y1 = y;
this._t0 = t1;
}
}
function MonotoneY(context) {
this._context = new ReflectContext(context);
}
(MonotoneY.prototype = Object.create(MonotoneX.prototype)).point = function(x, y) {
MonotoneX.prototype.point.call(this, y, x);
};
function ReflectContext(context) {
this._context = context;
}
ReflectContext.prototype = {
moveTo: function(x, y) { this._context.moveTo(y, x); },
closePath: function() { this._context.closePath(); },
lineTo: function(x, y) { this._context.lineTo(y, x); },
bezierCurveTo: function(x1, y1, x2, y2, x, y) { this._context.bezierCurveTo(y1, x1, y2, x2, y, x); }
};
function monotoneX(context) {
return new MonotoneX(context);
}
function monotoneY(context) {
return new MonotoneY(context);
}
function Natural(context) {
this._context = context;
}
Natural.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x = [];
this._y = [];
},
lineEnd: function() {
var x = this._x,
y = this._y,
n = x.length;
if (n) {
this._line ? this._context.lineTo(x[0], y[0]) : this._context.moveTo(x[0], y[0]);
if (n === 2) {
this._context.lineTo(x[1], y[1]);
} else {
var px = controlPoints(x),
py = controlPoints(y);
for (var i0 = 0, i1 = 1; i1 < n; ++i0, ++i1) {
this._context.bezierCurveTo(px[0][i0], py[0][i0], px[1][i0], py[1][i0], x[i1], y[i1]);
}
}
}
if (this._line || (this._line !== 0 && n === 1)) this._context.closePath();
this._line = 1 - this._line;
this._x = this._y = null;
},
point: function(x, y) {
this._x.push(+x);
this._y.push(+y);
}
};
// See https://www.particleincell.com/2012/bezier-splines/ for derivation.
function controlPoints(x) {
var i,
n = x.length - 1,
m,
a = new Array(n),
b = new Array(n),
r = new Array(n);
a[0] = 0, b[0] = 2, r[0] = x[0] + 2 * x[1];
for (i = 1; i < n - 1; ++i) a[i] = 1, b[i] = 4, r[i] = 4 * x[i] + 2 * x[i + 1];
a[n - 1] = 2, b[n - 1] = 7, r[n - 1] = 8 * x[n - 1] + x[n];
for (i = 1; i < n; ++i) m = a[i] / b[i - 1], b[i] -= m, r[i] -= m * r[i - 1];
a[n - 1] = r[n - 1] / b[n - 1];
for (i = n - 2; i >= 0; --i) a[i] = (r[i] - a[i + 1]) / b[i];
b[n - 1] = (x[n] + a[n - 1]) / 2;
for (i = 0; i < n - 1; ++i) b[i] = 2 * x[i + 1] - a[i + 1];
return [a, b];
}
function natural(context) {
return new Natural(context);
}
function Step(context, t) {
this._context = context;
this._t = t;
}
Step.prototype = {
areaStart: function() {
this._line = 0;
},
areaEnd: function() {
this._line = NaN;
},
lineStart: function() {
this._x = this._y = NaN;
this._point = 0;
},
lineEnd: function() {
if (0 < this._t && this._t < 1 && this._point === 2) this._context.lineTo(this._x, this._y);
if (this._line || (this._line !== 0 && this._point === 1)) this._context.closePath();
if (this._line >= 0) this._t = 1 - this._t, this._line = 1 - this._line;
},
point: function(x, y) {
x = +x, y = +y;
switch (this._point) {
case 0: this._point = 1; this._line ? this._context.lineTo(x, y) : this._context.moveTo(x, y); break;
case 1: this._point = 2; // proceed
default: {
if (this._t <= 0) {
this._context.lineTo(this._x, y);
this._context.lineTo(x, y);
} else {
var x1 = this._x * (1 - this._t) + x * this._t;
this._context.lineTo(x1, this._y);
this._context.lineTo(x1, y);
}
break;
}
}
this._x = x, this._y = y;
}
};
function step(context) {
return new Step(context, 0.5);
}
function stepBefore(context) {
return new Step(context, 0);
}
function stepAfter(context) {
return new Step(context, 1);
}
var slice$2 = Array.prototype.slice;
function none(series, order) {
if (!((n = series.length) > 1)) return;
for (var i = 1, s0, s1 = series[order[0]], n, m = s1.length; i < n; ++i) {
s0 = s1, s1 = series[order[i]];
for (var j = 0; j < m; ++j) {
s1[j][1] += s1[j][0] = isNaN(s0[j][1]) ? s0[j][0] : s0[j][1];
}
}
}
function none$1(series) {
var n = series.length, o = new Array(n);
while (--n >= 0) o[n] = n;
return o;
}
function stackValue(d, key) {
return d[key];
}
function stack() {
var keys = constant$1([]),
order = none$1,
offset = none,
value = stackValue;
function stack(data) {
var kz = keys.apply(this, arguments),
i,
m = data.length,
n = kz.length,
sz = new Array(n),
oz;
for (i = 0; i < n; ++i) {
for (var ki = kz[i], si = sz[i] = new Array(m), j = 0, sij; j < m; ++j) {
si[j] = sij = [0, +value(data[j], ki, j, data)];
sij.data = data[j];
}
si.key = ki;
}
for (i = 0, oz = order(sz); i < n; ++i) {
sz[oz[i]].index = i;
}
offset(sz, oz);
return sz;
}
stack.keys = function(_) {
return arguments.length ? (keys = typeof _ === "function" ? _ : constant$1(slice$2.call(_)), stack) : keys;
};
stack.value = function(_) {
return arguments.length ? (value = typeof _ === "function" ? _ : constant$1(+_), stack) : value;
};
stack.order = function(_) {
return arguments.length ? (order = _ == null ? none$1 : typeof _ === "function" ? _ : constant$1(slice$2.call(_)), stack) : order;
};
stack.offset = function(_) {
return arguments.length ? (offset = _ == null ? none : _, stack) : offset;
};
return stack;
}
function expand(series, order) {
if (!((n = series.length) > 0)) return;
for (var i, n, j = 0, m = series[0].length, y; j < m; ++j) {
for (y = i = 0; i < n; ++i) y += series[i][j][1] || 0;
if (y) for (i = 0; i < n; ++i) series[i][j][1] /= y;
}
none(series, order);
}
function silhouette(series, order) {
if (!((n = series.length) > 0)) return;
for (var j = 0, s0 = series[order[0]], n, m = s0.length; j < m; ++j) {
for (var i = 0, y = 0; i < n; ++i) y += series[i][j][1] || 0;
s0[j][1] += s0[j][0] = -y / 2;
}
none(series, order);
}
function wiggle(series, order) {
if (!((n = series.length) > 0) || !((m = (s0 = series[order[0]]).length) > 0)) return;
for (var y = 0, j = 1, s0, m, n; j < m; ++j) {
for (var i = 0, s1 = 0, s2 = 0; i < n; ++i) {
var si = series[order[i]],
sij0 = si[j][1] || 0,
sij1 = si[j - 1][1] || 0,
s3 = (sij0 - sij1) / 2;
for (var k = 0; k < i; ++k) {
var sk = series[order[k]],
skj0 = sk[j][1] || 0,
skj1 = sk[j - 1][1] || 0;
s3 += skj0 - skj1;
}
s1 += sij0, s2 += s3 * sij0;
}
s0[j - 1][1] += s0[j - 1][0] = y;
if (s1) y -= s2 / s1;
}
s0[j - 1][1] += s0[j - 1][0] = y;
none(series, order);
}
function ascending$1(series) {
var sums = series.map(sum$1);
return none$1(series).sort(function(a, b) { return sums[a] - sums[b]; });
}
function sum$1(series) {
var s = 0, i = -1, n = series.length, v;
while (++i < n) if (v = +series[i][1]) s += v;
return s;
}
function descending$2(series) {
return ascending$1(series).reverse();
}
function insideOut(series) {
var n = series.length,
i,
j,
sums = series.map(sum$1),
order = none$1(series).sort(function(a, b) { return sums[b] - sums[a]; }),
top = 0,
bottom = 0,
tops = [],
bottoms = [];
for (i = 0; i < n; ++i) {
j = order[i];
if (top < bottom) {
top += sums[j];
tops.push(j);
} else {
bottom += sums[j];
bottoms.push(j);
}
}
return bottoms.reverse().concat(tops);
}
function reverse(series) {
return none$1(series).reverse();
}
function define(constructor, factory, prototype) {
constructor.prototype = factory.prototype = prototype;
prototype.constructor = constructor;
}
function extend(parent, definition) {
var prototype = Object.create(parent.prototype);
for (var key in definition) prototype[key] = definition[key];
return prototype;
}
function Color() {}
var darker = 0.7;
var brighter = 1 / darker;
var reHex3 = /^#([0-9a-f]{3})$/;
var reHex6 = /^#([0-9a-f]{6})$/;
var reRgbInteger = /^rgb\(\s*([-+]?\d+)\s*,\s*([-+]?\d+)\s*,\s*([-+]?\d+)\s*\)$/;
var reRgbPercent = /^rgb\(\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*\)$/;
var reRgbaInteger = /^rgba\(\s*([-+]?\d+)\s*,\s*([-+]?\d+)\s*,\s*([-+]?\d+)\s*,\s*([-+]?\d+(?:\.\d+)?)\s*\)$/;
var reRgbaPercent = /^rgba\(\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)\s*\)$/;
var reHslPercent = /^hsl\(\s*([-+]?\d+(?:\.\d+)?)\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*\)$/;
var reHslaPercent = /^hsla\(\s*([-+]?\d+(?:\.\d+)?)\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)%\s*,\s*([-+]?\d+(?:\.\d+)?)\s*\)$/;
var named = {
aliceblue: 0xf0f8ff,
antiquewhite: 0xfaebd7,
aqua: 0x00ffff,
aquamarine: 0x7fffd4,
azure: 0xf0ffff,
beige: 0xf5f5dc,
bisque: 0xffe4c4,
black: 0x000000,
blanchedalmond: 0xffebcd,
blue: 0x0000ff,
blueviolet: 0x8a2be2,
brown: 0xa52a2a,
burlywood: 0xdeb887,
cadetblue: 0x5f9ea0,
chartreuse: 0x7fff00,
chocolate: 0xd2691e,
coral: 0xff7f50,
cornflowerblue: 0x6495ed,
cornsilk: 0xfff8dc,
crimson: 0xdc143c,
cyan: 0x00ffff,
darkblue: 0x00008b,
darkcyan: 0x008b8b,
darkgoldenrod: 0xb8860b,
darkgray: 0xa9a9a9,
darkgreen: 0x006400,
darkgrey: 0xa9a9a9,
darkkhaki: 0xbdb76b,
darkmagenta: 0x8b008b,
darkolivegreen: 0x556b2f,
darkorange: 0xff8c00,
darkorchid: 0x9932cc,
darkred: 0x8b0000,
darksalmon: 0xe9967a,
darkseagreen: 0x8fbc8f,
darkslateblue: 0x483d8b,
darkslategray: 0x2f4f4f,
darkslategrey: 0x2f4f4f,
darkturquoise: 0x00ced1,
darkviolet: 0x9400d3,
deeppink: 0xff1493,
deepskyblue: 0x00bfff,
dimgray: 0x696969,
dimgrey: 0x696969,
dodgerblue: 0x1e90ff,
firebrick: 0xb22222,
floralwhite: 0xfffaf0,
forestgreen: 0x228b22,
fuchsia: 0xff00ff,
gainsboro: 0xdcdcdc,
ghostwhite: 0xf8f8ff,
gold: 0xffd700,
goldenrod: 0xdaa520,
gray: 0x808080,
green: 0x008000,
greenyellow: 0xadff2f,
grey: 0x808080,
honeydew: 0xf0fff0,
hotpink: 0xff69b4,
indianred: 0xcd5c5c,
indigo: 0x4b0082,
ivory: 0xfffff0,
khaki: 0xf0e68c,
lavender: 0xe6e6fa,
lavenderblush: 0xfff0f5,
lawngreen: 0x7cfc00,
lemonchiffon: 0xfffacd,
lightblue: 0xadd8e6,
lightcoral: 0xf08080,
lightcyan: 0xe0ffff,
lightgoldenrodyellow: 0xfafad2,
lightgray: 0xd3d3d3,
lightgreen: 0x90ee90,
lightgrey: 0xd3d3d3,
lightpink: 0xffb6c1,
lightsalmon: 0xffa07a,
lightseagreen: 0x20b2aa,
lightskyblue: 0x87cefa,
lightslategray: 0x778899,
lightslategrey: 0x778899,
lightsteelblue: 0xb0c4de,
lightyellow: 0xffffe0,
lime: 0x00ff00,
limegreen: 0x32cd32,
linen: 0xfaf0e6,
magenta: 0xff00ff,
maroon: 0x800000,
mediumaquamarine: 0x66cdaa,
mediumblue: 0x0000cd,
mediumorchid: 0xba55d3,
mediumpurple: 0x9370db,
mediumseagreen: 0x3cb371,
mediumslateblue: 0x7b68ee,
mediumspringgreen: 0x00fa9a,
mediumturquoise: 0x48d1cc,
mediumvioletred: 0xc71585,
midnightblue: 0x191970,
mintcream: 0xf5fffa,
mistyrose: 0xffe4e1,
moccasin: 0xffe4b5,
navajowhite: 0xffdead,
navy: 0x000080,
oldlace: 0xfdf5e6,
olive: 0x808000,
olivedrab: 0x6b8e23,
orange: 0xffa500,
orangered: 0xff4500,
orchid: 0xda70d6,
palegoldenrod: 0xeee8aa,
palegreen: 0x98fb98,
paleturquoise: 0xafeeee,
palevioletred: 0xdb7093,
papayawhip: 0xffefd5,
peachpuff: 0xffdab9,
peru: 0xcd853f,
pink: 0xffc0cb,
plum: 0xdda0dd,
powderblue: 0xb0e0e6,
purple: 0x800080,
rebeccapurple: 0x663399,
red: 0xff0000,
rosybrown: 0xbc8f8f,
royalblue: 0x4169e1,
saddlebrown: 0x8b4513,
salmon: 0xfa8072,
sandybrown: 0xf4a460,
seagreen: 0x2e8b57,
seashell: 0xfff5ee,
sienna: 0xa0522d,
silver: 0xc0c0c0,
skyblue: 0x87ceeb,
slateblue: 0x6a5acd,
slategray: 0x708090,
slategrey: 0x708090,
snow: 0xfffafa,
springgreen: 0x00ff7f,
steelblue: 0x4682b4,
tan: 0xd2b48c,
teal: 0x008080,
thistle: 0xd8bfd8,
tomato: 0xff6347,
turquoise: 0x40e0d0,
violet: 0xee82ee,
wheat: 0xf5deb3,
white: 0xffffff,
whitesmoke: 0xf5f5f5,
yellow: 0xffff00,
yellowgreen: 0x9acd32
};
define(Color, color, {
displayable: function() {
return this.rgb().displayable();
},
toString: function() {
return this.rgb() + "";
}
});
function color(format) {
var m;
format = (format + "").trim().toLowerCase();
return (m = reHex3.exec(format)) ? (m = parseInt(m[1], 16), new Rgb((m >> 8 & 0xf) | (m >> 4 & 0x0f0), (m >> 4 & 0xf) | (m & 0xf0), ((m & 0xf) << 4) | (m & 0xf), 1)) // #f00
: (m = reHex6.exec(format)) ? rgbn(parseInt(m[1], 16)) // #ff0000
: (m = reRgbInteger.exec(format)) ? new Rgb(m[1], m[2], m[3], 1) // rgb(255, 0, 0)
: (m = reRgbPercent.exec(format)) ? new Rgb(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, 1) // rgb(100%, 0%, 0%)
: (m = reRgbaInteger.exec(format)) ? rgba(m[1], m[2], m[3], m[4]) // rgba(255, 0, 0, 1)
: (m = reRgbaPercent.exec(format)) ? rgba(m[1] * 255 / 100, m[2] * 255 / 100, m[3] * 255 / 100, m[4]) // rgb(100%, 0%, 0%, 1)
: (m = reHslPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, 1) // hsl(120, 50%, 50%)
: (m = reHslaPercent.exec(format)) ? hsla(m[1], m[2] / 100, m[3] / 100, m[4]) // hsla(120, 50%, 50%, 1)
: named.hasOwnProperty(format) ? rgbn(named[format])
: format === "transparent" ? new Rgb(NaN, NaN, NaN, 0)
: null;
}
function rgbn(n) {
return new Rgb(n >> 16 & 0xff, n >> 8 & 0xff, n & 0xff, 1);
}
function rgba(r, g, b, a) {
if (a <= 0) r = g = b = NaN;
return new Rgb(r, g, b, a);
}
function rgbConvert(o) {
if (!(o instanceof Color)) o = color(o);
if (!o) return new Rgb;
o = o.rgb();
return new Rgb(o.r, o.g, o.b, o.opacity);
}
function colorRgb(r, g, b, opacity) {
return arguments.length === 1 ? rgbConvert(r) : new Rgb(r, g, b, opacity == null ? 1 : opacity);
}
function Rgb(r, g, b, opacity) {
this.r = +r;
this.g = +g;
this.b = +b;
this.opacity = +opacity;
}
define(Rgb, colorRgb, extend(Color, {
brighter: function(k) {
k = k == null ? brighter : Math.pow(brighter, k);
return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
},
darker: function(k) {
k = k == null ? darker : Math.pow(darker, k);
return new Rgb(this.r * k, this.g * k, this.b * k, this.opacity);
},
rgb: function() {
return this;
},
displayable: function() {
return (0 <= this.r && this.r <= 255)
&& (0 <= this.g && this.g <= 255)
&& (0 <= this.b && this.b <= 255)
&& (0 <= this.opacity && this.opacity <= 1);
},
toString: function() {
var a = this.opacity; a = isNaN(a) ? 1 : Math.max(0, Math.min(1, a));
return (a === 1 ? "rgb(" : "rgba(")
+ Math.max(0, Math.min(255, Math.round(this.r) || 0)) + ", "
+ Math.max(0, Math.min(255, Math.round(this.g) || 0)) + ", "
+ Math.max(0, Math.min(255, Math.round(this.b) || 0))
+ (a === 1 ? ")" : ", " + a + ")");
}
}));
function hsla(h, s, l, a) {
if (a <= 0) h = s = l = NaN;
else if (l <= 0 || l >= 1) h = s = NaN;
else if (s <= 0) h = NaN;
return new Hsl(h, s, l, a);
}
function hslConvert(o) {
if (o instanceof Hsl) return new Hsl(o.h, o.s, o.l, o.opacity);
if (!(o instanceof Color)) o = color(o);
if (!o) return new Hsl;
if (o instanceof Hsl) return o;
o = o.rgb();
var r = o.r / 255,
g = o.g / 255,
b = o.b / 255,
min = Math.min(r, g, b),
max = Math.max(r, g, b),
h = NaN,
s = max - min,
l = (max + min) / 2;
if (s) {
if (r === max) h = (g - b) / s + (g < b) * 6;
else if (g === max) h = (b - r) / s + 2;
else h = (r - g) / s + 4;
s /= l < 0.5 ? max + min : 2 - max - min;
h *= 60;
} else {
s = l > 0 && l < 1 ? 0 : h;
}
return new Hsl(h, s, l, o.opacity);
}
function colorHsl(h, s, l, opacity) {
return arguments.length === 1 ? hslConvert(h) : new Hsl(h, s, l, opacity == null ? 1 : opacity);
}
function Hsl(h, s, l, opacity) {
this.h = +h;
this.s = +s;
this.l = +l;
this.opacity = +opacity;
}
define(Hsl, colorHsl, extend(Color, {
brighter: function(k) {
k = k == null ? brighter : Math.pow(brighter, k);
return new Hsl(this.h, this.s, this.l * k, this.opacity);
},
darker: function(k) {
k = k == null ? darker : Math.pow(darker, k);
return new Hsl(this.h, this.s, this.l * k, this.opacity);
},
rgb: function() {
var h = this.h % 360 + (this.h < 0) * 360,
s = isNaN(h) || isNaN(this.s) ? 0 : this.s,
l = this.l,
m2 = l + (l < 0.5 ? l : 1 - l) * s,
m1 = 2 * l - m2;
return new Rgb(
hsl2rgb(h >= 240 ? h - 240 : h + 120, m1, m2),
hsl2rgb(h, m1, m2),
hsl2rgb(h < 120 ? h + 240 : h - 120, m1, m2),
this.opacity
);
},
displayable: function() {
return (0 <= this.s && this.s <= 1 || isNaN(this.s))
&& (0 <= this.l && this.l <= 1)
&& (0 <= this.opacity && this.opacity <= 1);
}
}));
/* From FvD 13.37, CSS Color Module Level 3 */
function hsl2rgb(h, m1, m2) {
return (h < 60 ? m1 + (m2 - m1) * h / 60
: h < 180 ? m2
: h < 240 ? m1 + (m2 - m1) * (240 - h) / 60
: m1) * 255;
}
var deg2rad = Math.PI / 180;
var rad2deg = 180 / Math.PI;
var Kn = 18;
var Xn = 0.950470;
var Yn = 1;
var Zn = 1.088830;
var t0 = 4 / 29;
var t1 = 6 / 29;
var t2 = 3 * t1 * t1;
var t3 = t1 * t1 * t1;
function labConvert(o) {
if (o instanceof Lab) return new Lab(o.l, o.a, o.b, o.opacity);
if (o instanceof Hcl) {
var h = o.h * deg2rad;
return new Lab(o.l, Math.cos(h) * o.c, Math.sin(h) * o.c, o.opacity);
}
if (!(o instanceof Rgb)) o = rgbConvert(o);
var b = rgb2xyz(o.r),
a = rgb2xyz(o.g),
l = rgb2xyz(o.b),
x = xyz2lab((0.4124564 * b + 0.3575761 * a + 0.1804375 * l) / Xn),
y = xyz2lab((0.2126729 * b + 0.7151522 * a + 0.0721750 * l) / Yn),
z = xyz2lab((0.0193339 * b + 0.1191920 * a + 0.9503041 * l) / Zn);
return new Lab(116 * y - 16, 500 * (x - y), 200 * (y - z), o.opacity);
}
function lab(l, a, b, opacity) {
return arguments.length === 1 ? labConvert(l) : new Lab(l, a, b, opacity == null ? 1 : opacity);
}
function Lab(l, a, b, opacity) {
this.l = +l;
this.a = +a;
this.b = +b;
this.opacity = +opacity;
}
define(Lab, lab, extend(Color, {
brighter: function(k) {
return new Lab(this.l + Kn * (k == null ? 1 : k), this.a, this.b, this.opacity);
},
darker: function(k) {
return new Lab(this.l - Kn * (k == null ? 1 : k), this.a, this.b, this.opacity);
},
rgb: function() {
var y = (this.l + 16) / 116,
x = isNaN(this.a) ? y : y + this.a / 500,
z = isNaN(this.b) ? y : y - this.b / 200;
y = Yn * lab2xyz(y);
x = Xn * lab2xyz(x);
z = Zn * lab2xyz(z);
return new Rgb(
xyz2rgb( 3.2404542 * x - 1.5371385 * y - 0.4985314 * z), // D65 -> sRGB
xyz2rgb(-0.9692660 * x + 1.8760108 * y + 0.0415560 * z),
xyz2rgb( 0.0556434 * x - 0.2040259 * y + 1.0572252 * z),
this.opacity
);
}
}));
function xyz2lab(t) {
return t > t3 ? Math.pow(t, 1 / 3) : t / t2 + t0;
}
function lab2xyz(t) {
return t > t1 ? t * t * t : t2 * (t - t0);
}
function xyz2rgb(x) {
return 255 * (x <= 0.0031308 ? 12.92 * x : 1.055 * Math.pow(x, 1 / 2.4) - 0.055);
}
function rgb2xyz(x) {
return (x /= 255) <= 0.04045 ? x / 12.92 : Math.pow((x + 0.055) / 1.055, 2.4);
}
function hclConvert(o) {
if (o instanceof Hcl) return new Hcl(o.h, o.c, o.l, o.opacity);
if (!(o instanceof Lab)) o = labConvert(o);
var h = Math.atan2(o.b, o.a) * rad2deg;
return new Hcl(h < 0 ? h + 360 : h, Math.sqrt(o.a * o.a + o.b * o.b), o.l, o.opacity);
}
function colorHcl(h, c, l, opacity) {
return arguments.length === 1 ? hclConvert(h) : new Hcl(h, c, l, opacity == null ? 1 : opacity);
}
function Hcl(h, c, l, opacity) {
this.h = +h;
this.c = +c;
this.l = +l;
this.opacity = +opacity;
}
define(Hcl, colorHcl, extend(Color, {
brighter: function(k) {
return new Hcl(this.h, this.c, this.l + Kn * (k == null ? 1 : k), this.opacity);
},
darker: function(k) {
return new Hcl(this.h, this.c, this.l - Kn * (k == null ? 1 : k), this.opacity);
},
rgb: function() {
return labConvert(this).rgb();
}
}));
var A = -0.14861;
var B = +1.78277;
var C = -0.29227;
var D = -0.90649;
var E = +1.97294;
var ED = E * D;
var EB = E * B;
var BC_DA = B * C - D * A;
function cubehelixConvert(o) {
if (o instanceof Cubehelix) return new Cubehelix(o.h, o.s, o.l, o.opacity);
if (!(o instanceof Rgb)) o = rgbConvert(o);
var r = o.r / 255,
g = o.g / 255,
b = o.b / 255,
l = (BC_DA * b + ED * r - EB * g) / (BC_DA + ED - EB),
bl = b - l,
k = (E * (g - l) - C * bl) / D,
s = Math.sqrt(k * k + bl * bl) / (E * l * (1 - l)), // NaN if l=0 or l=1
h = s ? Math.atan2(k, bl) * rad2deg - 120 : NaN;
return new Cubehelix(h < 0 ? h + 360 : h, s, l, o.opacity);
}
function cubehelix(h, s, l, opacity) {
return arguments.length === 1 ? cubehelixConvert(h) : new Cubehelix(h, s, l, opacity == null ? 1 : opacity);
}
function Cubehelix(h, s, l, opacity) {
this.h = +h;
this.s = +s;
this.l = +l;
this.opacity = +opacity;
}
define(Cubehelix, cubehelix, extend(Color, {
brighter: function(k) {
k = k == null ? brighter : Math.pow(brighter, k);
return new Cubehelix(this.h, this.s, this.l * k, this.opacity);
},
darker: function(k) {
k = k == null ? darker : Math.pow(darker, k);
return new Cubehelix(this.h, this.s, this.l * k, this.opacity);
},
rgb: function() {
var h = isNaN(this.h) ? 0 : (this.h + 120) * deg2rad,
l = +this.l,
a = isNaN(this.s) ? 0 : this.s * l * (1 - l),
cosh = Math.cos(h),
sinh = Math.sin(h);
return new Rgb(
255 * (l + a * (A * cosh + B * sinh)),
255 * (l + a * (C * cosh + D * sinh)),
255 * (l + a * (E * cosh)),
this.opacity
);
}
}));
function basis$1(t1, v0, v1, v2, v3) {
var t2 = t1 * t1, t3 = t2 * t1;
return ((1 - 3 * t1 + 3 * t2 - t3) * v0
+ (4 - 6 * t2 + 3 * t3) * v1
+ (1 + 3 * t1 + 3 * t2 - 3 * t3) * v2
+ t3 * v3) / 6;
}
function basis$2(values) {
var n = values.length - 1;
return function(t) {
var i = t <= 0 ? (t = 0) : t >= 1 ? (t = 1, n - 1) : Math.floor(t * n),
v1 = values[i],
v2 = values[i + 1],
v0 = i > 0 ? values[i - 1] : 2 * v1 - v2,
v3 = i < n - 1 ? values[i + 2] : 2 * v2 - v1;
return basis$1((t - i / n) * n, v0, v1, v2, v3);
};
}
function basisClosed$1(values) {
var n = values.length;
return function(t) {
var i = Math.floor(((t %= 1) < 0 ? ++t : t) * n),
v0 = values[(i + n - 1) % n],
v1 = values[i % n],
v2 = values[(i + 1) % n],
v3 = values[(i + 2) % n];
return basis$1((t - i / n) * n, v0, v1, v2, v3);
};
}
function constant$2(x) {
return function() {
return x;
};
}
function linear$1(a, d) {
return function(t) {
return a + t * d;
};
}
function exponential$1(a, b, y) {
return a = Math.pow(a, y), b = Math.pow(b, y) - a, y = 1 / y, function(t) {
return Math.pow(a + t * b, y);
};
}
function hue(a, b) {
var d = b - a;
return d ? linear$1(a, d > 180 || d < -180 ? d - 360 * Math.round(d / 360) : d) : constant$2(isNaN(a) ? b : a);
}
function gamma(y) {
return (y = +y) === 1 ? nogamma : function(a, b) {
return b - a ? exponential$1(a, b, y) : constant$2(isNaN(a) ? b : a);
};
}
function nogamma(a, b) {
var d = b - a;
return d ? linear$1(a, d) : constant$2(isNaN(a) ? b : a);
}
var interpolateRgb = (function rgbGamma(y) {
var color = gamma(y);
function rgb(start, end) {
var r = color((start = colorRgb(start)).r, (end = colorRgb(end)).r),
g = color(start.g, end.g),
b = color(start.b, end.b),
opacity = color(start.opacity, end.opacity);
return function(t) {
start.r = r(t);
start.g = g(t);
start.b = b(t);
start.opacity = opacity(t);
return start + "";
};
}
rgb.gamma = rgbGamma;
return rgb;
})(1);
function rgbSpline(spline) {
return function(colors) {
var n = colors.length,
r = new Array(n),
g = new Array(n),
b = new Array(n),
i, color;
for (i = 0; i < n; ++i) {
color = colorRgb(colors[i]);
r[i] = color.r || 0;
g[i] = color.g || 0;
b[i] = color.b || 0;
}
r = spline(r);
g = spline(g);
b = spline(b);
color.opacity = 1;
return function(t) {
color.r = r(t);
color.g = g(t);
color.b = b(t);
return color + "";
};
};
}
var rgbBasis = rgbSpline(basis$2);
var rgbBasisClosed = rgbSpline(basisClosed$1);
function array$1(a, b) {
var nb = b ? b.length : 0,
na = a ? Math.min(nb, a.length) : 0,
x = new Array(nb),
c = new Array(nb),
i;
for (i = 0; i < na; ++i) x[i] = interpolate(a[i], b[i]);
for (; i < nb; ++i) c[i] = b[i];
return function(t) {
for (i = 0; i < na; ++i) c[i] = x[i](t);
return c;
};
}
function date(a, b) {
var d = new Date;
return a = +a, b -= a, function(t) {
return d.setTime(a + b * t), d;
};
}
function interpolateNumber(a, b) {
return a = +a, b -= a, function(t) {
return a + b * t;
};
}
function object(a, b) {
var i = {},
c = {},
k;
if (a === null || typeof a !== "object") a = {};
if (b === null || typeof b !== "object") b = {};
for (k in b) {
if (k in a) {
i[k] = interpolate(a[k], b[k]);
} else {
c[k] = b[k];
}
}
return function(t) {
for (k in i) c[k] = i[k](t);
return c;
};
}
var reA = /[-+]?(?:\d+\.?\d*|\.?\d+)(?:[eE][-+]?\d+)?/g;
var reB = new RegExp(reA.source, "g");
function zero(b) {
return function() {
return b;
};
}
function one(b) {
return function(t) {
return b(t) + "";
};
}
function interpolateString(a, b) {
var bi = reA.lastIndex = reB.lastIndex = 0, // scan index for next number in b
am, // current match in a
bm, // current match in b
bs, // string preceding current number in b, if any
i = -1, // index in s
s = [], // string constants and placeholders
q = []; // number interpolators
// Coerce inputs to strings.
a = a + "", b = b + "";
// Interpolate pairs of numbers in a & b.
while ((am = reA.exec(a))
&& (bm = reB.exec(b))) {
if ((bs = bm.index) > bi) { // a string precedes the next number in b
bs = b.slice(bi, bs);
if (s[i]) s[i] += bs; // coalesce with previous string
else s[++i] = bs;
}
if ((am = am[0]) === (bm = bm[0])) { // numbers in a & b match
if (s[i]) s[i] += bm; // coalesce with previous string
else s[++i] = bm;
} else { // interpolate non-matching numbers
s[++i] = null;
q.push({i: i, x: interpolateNumber(am, bm)});
}
bi = reB.lastIndex;
}
// Add remains of b.
if (bi < b.length) {
bs = b.slice(bi);
if (s[i]) s[i] += bs; // coalesce with previous string
else s[++i] = bs;
}
// Special optimization for only a single match.
// Otherwise, interpolate each of the numbers and rejoin the string.
return s.length < 2 ? (q[0]
? one(q[0].x)
: zero(b))
: (b = q.length, function(t) {
for (var i = 0, o; i < b; ++i) s[(o = q[i]).i] = o.x(t);
return s.join("");
});
}
function interpolate(a, b) {
var t = typeof b, c;
return b == null || t === "boolean" ? constant$2(b)
: (t === "number" ? interpolateNumber
: t === "string" ? ((c = color(b)) ? (b = c, interpolateRgb) : interpolateString)
: b instanceof color ? interpolateRgb
: b instanceof Date ? date
: Array.isArray(b) ? array$1
: isNaN(b) ? object
: interpolateNumber)(a, b);
}
function interpolateRound(a, b) {
return a = +a, b -= a, function(t) {
return Math.round(a + b * t);
};
}
var degrees = 180 / Math.PI;
var identity$2 = {
translateX: 0,
translateY: 0,
rotate: 0,
skewX: 0,
scaleX: 1,
scaleY: 1
};
function decompose(a, b, c, d, e, f) {
var scaleX, scaleY, skewX;
if (scaleX = Math.sqrt(a * a + b * b)) a /= scaleX, b /= scaleX;
if (skewX = a * c + b * d) c -= a * skewX, d -= b * skewX;
if (scaleY = Math.sqrt(c * c + d * d)) c /= scaleY, d /= scaleY, skewX /= scaleY;
if (a * d < b * c) a = -a, b = -b, skewX = -skewX, scaleX = -scaleX;
return {
translateX: e,
translateY: f,
rotate: Math.atan2(b, a) * degrees,
skewX: Math.atan(skewX) * degrees,
scaleX: scaleX,
scaleY: scaleY
};
}
var cssNode;
var cssRoot;
var cssView;
var svgNode;
function parseCss(value) {
if (value === "none") return identity$2;
if (!cssNode) cssNode = document.createElement("DIV"), cssRoot = document.documentElement, cssView = document.defaultView;
cssNode.style.transform = value;
value = cssView.getComputedStyle(cssRoot.appendChild(cssNode), null).getPropertyValue("transform");
cssRoot.removeChild(cssNode);
value = value.slice(7, -1).split(",");
return decompose(+value[0], +value[1], +value[2], +value[3], +value[4], +value[5]);
}
function parseSvg(value) {
if (value == null) return identity$2;
if (!svgNode) svgNode = document.createElementNS("http://www.w3.org/2000/svg", "g");
svgNode.setAttribute("transform", value);
if (!(value = svgNode.transform.baseVal.consolidate())) return identity$2;
value = value.matrix;
return decompose(value.a, value.b, value.c, value.d, value.e, value.f);
}
function interpolateTransform(parse, pxComma, pxParen, degParen) {
function pop(s) {
return s.length ? s.pop() + " " : "";
}
function translate(xa, ya, xb, yb, s, q) {
if (xa !== xb || ya !== yb) {
var i = s.push("translate(", null, pxComma, null, pxParen);
q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
} else if (xb || yb) {
s.push("translate(" + xb + pxComma + yb + pxParen);
}
}
function rotate(a, b, s, q) {
if (a !== b) {
if (a - b > 180) b += 360; else if (b - a > 180) a += 360; // shortest path
q.push({i: s.push(pop(s) + "rotate(", null, degParen) - 2, x: interpolateNumber(a, b)});
} else if (b) {
s.push(pop(s) + "rotate(" + b + degParen);
}
}
function skewX(a, b, s, q) {
if (a !== b) {
q.push({i: s.push(pop(s) + "skewX(", null, degParen) - 2, x: interpolateNumber(a, b)});
} else if (b) {
s.push(pop(s) + "skewX(" + b + degParen);
}
}
function scale(xa, ya, xb, yb, s, q) {
if (xa !== xb || ya !== yb) {
var i = s.push(pop(s) + "scale(", null, ",", null, ")");
q.push({i: i - 4, x: interpolateNumber(xa, xb)}, {i: i - 2, x: interpolateNumber(ya, yb)});
} else if (xb !== 1 || yb !== 1) {
s.push(pop(s) + "scale(" + xb + "," + yb + ")");
}
}
return function(a, b) {
var s = [], // string constants and placeholders
q = []; // number interpolators
a = parse(a), b = parse(b);
translate(a.translateX, a.translateY, b.translateX, b.translateY, s, q);
rotate(a.rotate, b.rotate, s, q);
skewX(a.skewX, b.skewX, s, q);
scale(a.scaleX, a.scaleY, b.scaleX, b.scaleY, s, q);
a = b = null; // gc
return function(t) {
var i = -1, n = q.length, o;
while (++i < n) s[(o = q[i]).i] = o.x(t);
return s.join("");
};
};
}
var interpolateTransform$1 = interpolateTransform(parseCss, "px, ", "px)", "deg)");
var interpolateTransform$2 = interpolateTransform(parseSvg, ", ", ")", ")");
var rho = Math.SQRT2;
var rho2 = 2;
var rho4 = 4;
var epsilon2 = 1e-12;
function cosh(x) {
return ((x = Math.exp(x)) + 1 / x) / 2;
}
function sinh(x) {
return ((x = Math.exp(x)) - 1 / x) / 2;
}
function tanh(x) {
return ((x = Math.exp(2 * x)) - 1) / (x + 1);
}
// p0 = [ux0, uy0, w0]
// p1 = [ux1, uy1, w1]
function interpolateZoom(p0, p1) {
var ux0 = p0[0], uy0 = p0[1], w0 = p0[2],
ux1 = p1[0], uy1 = p1[1], w1 = p1[2],
dx = ux1 - ux0,
dy = uy1 - uy0,
d2 = dx * dx + dy * dy,
i,
S;
// Special case for u0 ≅ u1.
if (d2 < epsilon2) {
S = Math.log(w1 / w0) / rho;
i = function(t) {
return [
ux0 + t * dx,
uy0 + t * dy,
w0 * Math.exp(rho * t * S)
];
}
}
// General case.
else {
var d1 = Math.sqrt(d2),
b0 = (w1 * w1 - w0 * w0 + rho4 * d2) / (2 * w0 * rho2 * d1),
b1 = (w1 * w1 - w0 * w0 - rho4 * d2) / (2 * w1 * rho2 * d1),
r0 = Math.log(Math.sqrt(b0 * b0 + 1) - b0),
r1 = Math.log(Math.sqrt(b1 * b1 + 1) - b1);
S = (r1 - r0) / rho;
i = function(t) {
var s = t * S,
coshr0 = cosh(r0),
u = w0 / (rho2 * d1) * (coshr0 * tanh(rho * s + r0) - sinh(r0));
return [
ux0 + u * dx,
uy0 + u * dy,
w0 * coshr0 / cosh(rho * s + r0)
];
}
}
i.duration = S * 1000;
return i;
}
function hsl(hue) {
return function(start, end) {
var h = hue((start = colorHsl(start)).h, (end = colorHsl(end)).h),
s = nogamma(start.s, end.s),
l = nogamma(start.l, end.l),
opacity = nogamma(start.opacity, end.opacity);
return function(t) {
start.h = h(t);
start.s = s(t);
start.l = l(t);
start.opacity = opacity(t);
return start + "";
};
}
}
var hsl$1 = hsl(hue);
var hslLong = hsl(nogamma);
function lab$1(start, end) {
var l = nogamma((start = lab(start)).l, (end = lab(end)).l),
a = nogamma(start.a, end.a),
b = nogamma(start.b, end.b),
opacity = nogamma(start.opacity, end.opacity);
return function(t) {
start.l = l(t);
start.a = a(t);
start.b = b(t);
start.opacity = opacity(t);
return start + "";
};
}
function hcl(hue) {
return function(start, end) {
var h = hue((start = colorHcl(start)).h, (end = colorHcl(end)).h),
c = nogamma(start.c, end.c),
l = nogamma(start.l, end.l),
opacity = nogamma(start.opacity, end.opacity);
return function(t) {
start.h = h(t);
start.c = c(t);
start.l = l(t);
start.opacity = opacity(t);
return start + "";
};
}
}
var hcl$1 = hcl(hue);
var hclLong = hcl(nogamma);
function cubehelix$1(hue) {
return (function cubehelixGamma(y) {
y = +y;
function cubehelix$$(start, end) {
var h = hue((start = cubehelix(start)).h, (end = cubehelix(end)).h),
s = nogamma(start.s, end.s),
l = nogamma(start.l, end.l),
opacity = nogamma(start.opacity, end.opacity);
return function(t) {
start.h = h(t);
start.s = s(t);
start.l = l(Math.pow(t, y));
start.opacity = opacity(t);
return start + "";
};
}
cubehelix$$.gamma = cubehelixGamma;
return cubehelix$$;
})(1);
}
var cubehelix$2 = cubehelix$1(hue);
var interpolateCubehelixLong = cubehelix$1(nogamma);
function quantize(interpolator, n) {
var samples = new Array(n);
for (var i = 0; i < n; ++i) samples[i] = interpolator(i / (n - 1));
return samples;
}
var noop$1 = {value: function() {}};
function dispatch() {
for (var i = 0, n = arguments.length, _ = {}, t; i < n; ++i) {
if (!(t = arguments[i] + "") || (t in _)) throw new Error("illegal type: " + t);
_[t] = [];
}
return new Dispatch(_);
}
function Dispatch(_) {
this._ = _;
}
function parseTypenames(typenames, types) {
return typenames.trim().split(/^|\s+/).map(function(t) {
var name = "", i = t.indexOf(".");
if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
if (t && !types.hasOwnProperty(t)) throw new Error("unknown type: " + t);
return {type: t, name: name};
});
}
Dispatch.prototype = dispatch.prototype = {
constructor: Dispatch,
on: function(typename, callback) {
var _ = this._,
T = parseTypenames(typename + "", _),
t,
i = -1,
n = T.length;
// If no callback was specified, return the callback of the given type and name.
if (arguments.length < 2) {
while (++i < n) if ((t = (typename = T[i]).type) && (t = get(_[t], typename.name))) return t;
return;
}
// If a type was specified, set the callback for the given type and name.
// Otherwise, if a null callback was specified, remove callbacks of the given name.
if (callback != null && typeof callback !== "function") throw new Error("invalid callback: " + callback);
while (++i < n) {
if (t = (typename = T[i]).type) _[t] = set$1(_[t], typename.name, callback);
else if (callback == null) for (t in _) _[t] = set$1(_[t], typename.name, null);
}
return this;
},
copy: function() {
var copy = {}, _ = this._;
for (var t in _) copy[t] = _[t].slice();
return new Dispatch(copy);
},
call: function(type, that) {
if ((n = arguments.length - 2) > 0) for (var args = new Array(n), i = 0, n, t; i < n; ++i) args[i] = arguments[i + 2];
if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
for (t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
},
apply: function(type, that, args) {
if (!this._.hasOwnProperty(type)) throw new Error("unknown type: " + type);
for (var t = this._[type], i = 0, n = t.length; i < n; ++i) t[i].value.apply(that, args);
}
};
function get(type, name) {
for (var i = 0, n = type.length, c; i < n; ++i) {
if ((c = type[i]).name === name) {
return c.value;
}
}
}
function set$1(type, name, callback) {
for (var i = 0, n = type.length; i < n; ++i) {
if (type[i].name === name) {
type[i] = noop$1, type = type.slice(0, i).concat(type.slice(i + 1));
break;
}
}
if (callback != null) type.push({name: name, value: callback});
return type;
}
function objectConverter(columns) {
return new Function("d", "return {" + columns.map(function(name, i) {
return JSON.stringify(name) + ": d[" + i + "]";
}).join(",") + "}");
}
function customConverter(columns, f) {
var object = objectConverter(columns);
return function(row, i) {
return f(object(row), i, columns);
};
}
// Compute unique columns in order of discovery.
function inferColumns(rows) {
var columnSet = Object.create(null),
columns = [];
rows.forEach(function(row) {
for (var column in row) {
if (!(column in columnSet)) {
columns.push(columnSet[column] = column);
}
}
});
return columns;
}
function dsv(delimiter) {
var reFormat = new RegExp("[\"" + delimiter + "\n]"),
delimiterCode = delimiter.charCodeAt(0);
function parse(text, f) {
var convert, columns, rows = parseRows(text, function(row, i) {
if (convert) return convert(row, i - 1);
columns = row, convert = f ? customConverter(row, f) : objectConverter(row);
});
rows.columns = columns;
return rows;
}
function parseRows(text, f) {
var EOL = {}, // sentinel value for end-of-line
EOF = {}, // sentinel value for end-of-file
rows = [], // output rows
N = text.length,
I = 0, // current character index
n = 0, // the current line number
t, // the current token
eol; // is the current token followed by EOL?
function token() {
if (I >= N) return EOF; // special case: end of file
if (eol) return eol = false, EOL; // special case: end of line
// special case: quotes
var j = I, c;
if (text.charCodeAt(j) === 34) {
var i = j;
while (i++ < N) {
if (text.charCodeAt(i) === 34) {
if (text.charCodeAt(i + 1) !== 34) break;
++i;
}
}
I = i + 2;
c = text.charCodeAt(i + 1);
if (c === 13) {
eol = true;
if (text.charCodeAt(i + 2) === 10) ++I;
} else if (c === 10) {
eol = true;
}
return text.slice(j + 1, i).replace(/""/g, "\"");
}
// common case: find next delimiter or newline
while (I < N) {
var k = 1;
c = text.charCodeAt(I++);
if (c === 10) eol = true; // \n
else if (c === 13) { eol = true; if (text.charCodeAt(I) === 10) ++I, ++k; } // \r|\r\n
else if (c !== delimiterCode) continue;
return text.slice(j, I - k);
}
// special case: last token before EOF
return text.slice(j);
}
while ((t = token()) !== EOF) {
var a = [];
while (t !== EOL && t !== EOF) {
a.push(t);
t = token();
}
if (f && (a = f(a, n++)) == null) continue;
rows.push(a);
}
return rows;
}
function format(rows, columns) {
if (columns == null) columns = inferColumns(rows);
return [columns.map(formatValue).join(delimiter)].concat(rows.map(function(row) {
return columns.map(function(column) {
return formatValue(row[column]);
}).join(delimiter);
})).join("\n");
}
function formatRows(rows) {
return rows.map(formatRow).join("\n");
}
function formatRow(row) {
return row.map(formatValue).join(delimiter);
}
function formatValue(text) {
return text == null ? ""
: reFormat.test(text += "") ? "\"" + text.replace(/\"/g, "\"\"") + "\""
: text;
}
return {
parse: parse,
parseRows: parseRows,
format: format,
formatRows: formatRows
};
}
var csv = dsv(",");
var csvParse = csv.parse;
var csvParseRows = csv.parseRows;
var csvFormat = csv.format;
var csvFormatRows = csv.formatRows;
var tsv = dsv("\t");
var tsvParse = tsv.parse;
var tsvParseRows = tsv.parseRows;
var tsvFormat = tsv.format;
var tsvFormatRows = tsv.formatRows;
function request(url, callback) {
var request,
event = dispatch("beforesend", "progress", "load", "error"),
mimeType,
headers = map$1(),
xhr = new XMLHttpRequest,
user = null,
password = null,
response,
responseType,
timeout = 0;
// If IE does not support CORS, use XDomainRequest.
if (typeof XDomainRequest !== "undefined"
&& !("withCredentials" in xhr)
&& /^(http(s)?:)?\/\//.test(url)) xhr = new XDomainRequest;
"onload" in xhr
? xhr.onload = xhr.onerror = xhr.ontimeout = respond
: xhr.onreadystatechange = function(o) { xhr.readyState > 3 && respond(o); };
function respond(o) {
var status = xhr.status, result;
if (!status && hasResponse(xhr)
|| status >= 200 && status < 300
|| status === 304) {
if (response) {
try {
result = response.call(request, xhr);
} catch (e) {
event.call("error", request, e);
return;
}
} else {
result = xhr;
}
event.call("load", request, result);
} else {
event.call("error", request, o);
}
}
xhr.onprogress = function(e) {
event.call("progress", request, e);
};
request = {
header: function(name, value) {
name = (name + "").toLowerCase();
if (arguments.length < 2) return headers.get(name);
if (value == null) headers.remove(name);
else headers.set(name, value + "");
return request;
},
// If mimeType is non-null and no Accept header is set, a default is used.
mimeType: function(value) {
if (!arguments.length) return mimeType;
mimeType = value == null ? null : value + "";
return request;
},
// Specifies what type the response value should take;
// for instance, arraybuffer, blob, document, or text.
responseType: function(value) {
if (!arguments.length) return responseType;
responseType = value;
return request;
},
timeout: function(value) {
if (!arguments.length) return timeout;
timeout = +value;
return request;
},
user: function(value) {
return arguments.length < 1 ? user : (user = value == null ? null : value + "", request);
},
password: function(value) {
return arguments.length < 1 ? password : (password = value == null ? null : value + "", request);
},
// Specify how to convert the response content to a specific type;
// changes the callback value on "load" events.
response: function(value) {
response = value;
return request;
},
// Alias for send("GET", …).
get: function(data, callback) {
return request.send("GET", data, callback);
},
// Alias for send("POST", …).
post: function(data, callback) {
return request.send("POST", data, callback);
},
// If callback is non-null, it will be used for error and load events.
send: function(method, data, callback) {
xhr.open(method, url, true, user, password);
if (mimeType != null && !headers.has("accept")) headers.set("accept", mimeType + ",*/*");
if (xhr.setRequestHeader) headers.each(function(value, name) { xhr.setRequestHeader(name, value); });
if (mimeType != null && xhr.overrideMimeType) xhr.overrideMimeType(mimeType);
if (responseType != null) xhr.responseType = responseType;
if (timeout > 0) xhr.timeout = timeout;
if (callback == null && typeof data === "function") callback = data, data = null;
if (callback != null && callback.length === 1) callback = fixCallback(callback);
if (callback != null) request.on("error", callback).on("load", function(xhr) { callback(null, xhr); });
event.call("beforesend", request, xhr);
xhr.send(data == null ? null : data);
return request;
},
abort: function() {
xhr.abort();
return request;
},
on: function() {
var value = event.on.apply(event, arguments);
return value === event ? request : value;
}
};
if (callback != null) {
if (typeof callback !== "function") throw new Error("invalid callback: " + callback);
return request.get(callback);
}
return request;
}
function fixCallback(callback) {
return function(error, xhr) {
callback(error == null ? xhr : null);
};
}
function hasResponse(xhr) {
var type = xhr.responseType;
return type && type !== "text"
? xhr.response // null on error
: xhr.responseText; // "" on error
}
function type(defaultMimeType, response) {
return function(url, callback) {
var r = request(url).mimeType(defaultMimeType).response(response);
if (callback != null) {
if (typeof callback !== "function") throw new Error("invalid callback: " + callback);
return r.get(callback);
}
return r;
};
}
var html = type("text/html", function(xhr) {
return document.createRange().createContextualFragment(xhr.responseText);
});
var json = type("application/json", function(xhr) {
return JSON.parse(xhr.responseText);
});
var text = type("text/plain", function(xhr) {
return xhr.responseText;
});
var xml = type("application/xml", function(xhr) {
var xml = xhr.responseXML;
if (!xml) throw new Error("parse error");
return xml;
});
function dsv$1(defaultMimeType, parse) {
return function(url, row, callback) {
if (arguments.length < 3) callback = row, row = null;
var r = request(url).mimeType(defaultMimeType);
r.row = function(_) { return arguments.length ? r.response(responseOf(parse, row = _)) : row; };
r.row(row);
return callback ? r.get(callback) : r;
};
}
function responseOf(parse, row) {
return function(request) {
return parse(request.responseText, row);
};
}
var csv$1 = dsv$1("text/csv", csvParse);
var tsv$1 = dsv$1("text/tab-separated-values", tsvParse);
var frame = 0;
var timeout = 0;
var interval = 0;
var pokeDelay = 1000;
var taskHead;
var taskTail;
var clockLast = 0;
var clockNow = 0;
var clockSkew = 0;
var clock = typeof performance === "object" && performance.now ? performance : Date;
var setFrame = typeof requestAnimationFrame === "function"
? (clock === Date ? function(f) { requestAnimationFrame(function() { f(clock.now()); }); } : requestAnimationFrame)
: function(f) { setTimeout(f, 17); };
function now() {
return clockNow || (setFrame(clearNow), clockNow = clock.now() + clockSkew);
}
function clearNow() {
clockNow = 0;
}
function Timer() {
this._call =
this._time =
this._next = null;
}
Timer.prototype = timer.prototype = {
constructor: Timer,
restart: function(callback, delay, time) {
if (typeof callback !== "function") throw new TypeError("callback is not a function");
time = (time == null ? now() : +time) + (delay == null ? 0 : +delay);
if (!this._next && taskTail !== this) {
if (taskTail) taskTail._next = this;
else taskHead = this;
taskTail = this;
}
this._call = callback;
this._time = time;
sleep();
},
stop: function() {
if (this._call) {
this._call = null;
this._time = Infinity;
sleep();
}
}
};
function timer(callback, delay, time) {
var t = new Timer;
t.restart(callback, delay, time);
return t;
}
function timerFlush() {
now(); // Get the current time, if not already set.
++frame; // Pretend we’ve set an alarm, if we haven’t already.
var t = taskHead, e;
while (t) {
if ((e = clockNow - t._time) >= 0) t._call.call(null, e);
t = t._next;
}
--frame;
}
function wake(time) {
clockNow = (clockLast = time || clock.now()) + clockSkew;
frame = timeout = 0;
try {
timerFlush();
} finally {
frame = 0;
nap();
clockNow = 0;
}
}
function poke$1() {
var now = clock.now(), delay = now - clockLast;
if (delay > pokeDelay) clockSkew -= delay, clockLast = now;
}
function nap() {
var t0, t1 = taskHead, t2, time = Infinity;
while (t1) {
if (t1._call) {
if (time > t1._time) time = t1._time;
t0 = t1, t1 = t1._next;
} else {
t2 = t1._next, t1._next = null;
t1 = t0 ? t0._next = t2 : taskHead = t2;
}
}
taskTail = t0;
sleep(time);
}
function sleep(time) {
if (frame) return; // Soonest alarm already set, or will be.
if (timeout) timeout = clearTimeout(timeout);
var delay = time - clockNow;
if (delay > 24) {
if (time < Infinity) timeout = setTimeout(wake, delay);
if (interval) interval = clearInterval(interval);
} else {
if (!interval) interval = setInterval(poke$1, pokeDelay);
frame = 1, setFrame(wake);
}
}
function timeout$1(callback, delay, time) {
var t = new Timer;
delay = delay == null ? 0 : +delay;
t.restart(function(elapsed) {
t.stop();
callback(elapsed + delay);
}, delay, time);
return t;
}
function interval$1(callback, delay, time) {
var t = new Timer, total = delay;
if (delay == null) return t.restart(callback, delay, time), t;
delay = +delay, time = time == null ? now() : +time;
t.restart(function tick(elapsed) {
elapsed += total;
t.restart(tick, total += delay, time);
callback(elapsed);
}, delay, time);
return t;
}
var t0$1 = new Date;
var t1$1 = new Date;
function newInterval(floori, offseti, count, field) {
function interval(date) {
return floori(date = new Date(+date)), date;
}
interval.floor = interval;
interval.ceil = function(date) {
return floori(date = new Date(date - 1)), offseti(date, 1), floori(date), date;
};
interval.round = function(date) {
var d0 = interval(date),
d1 = interval.ceil(date);
return date - d0 < d1 - date ? d0 : d1;
};
interval.offset = function(date, step) {
return offseti(date = new Date(+date), step == null ? 1 : Math.floor(step)), date;
};
interval.range = function(start, stop, step) {
var range = [];
start = interval.ceil(start);
step = step == null ? 1 : Math.floor(step);
if (!(start < stop) || !(step > 0)) return range; // also handles Invalid Date
do range.push(new Date(+start)); while (offseti(start, step), floori(start), start < stop)
return range;
};
interval.filter = function(test) {
return newInterval(function(date) {
while (floori(date), !test(date)) date.setTime(date - 1);
}, function(date, step) {
while (--step >= 0) while (offseti(date, 1), !test(date));
});
};
if (count) {
interval.count = function(start, end) {
t0$1.setTime(+start), t1$1.setTime(+end);
floori(t0$1), floori(t1$1);
return Math.floor(count(t0$1, t1$1));
};
interval.every = function(step) {
step = Math.floor(step);
return !isFinite(step) || !(step > 0) ? null
: !(step > 1) ? interval
: interval.filter(field
? function(d) { return field(d) % step === 0; }
: function(d) { return interval.count(0, d) % step === 0; });
};
}
return interval;
}
var millisecond = newInterval(function() {
// noop
}, function(date, step) {
date.setTime(+date + step);
}, function(start, end) {
return end - start;
});
// An optimized implementation for this simple case.
millisecond.every = function(k) {
k = Math.floor(k);
if (!isFinite(k) || !(k > 0)) return null;
if (!(k > 1)) return millisecond;
return newInterval(function(date) {
date.setTime(Math.floor(date / k) * k);
}, function(date, step) {
date.setTime(+date + step * k);
}, function(start, end) {
return (end - start) / k;
});
};
var milliseconds = millisecond.range;
var durationSecond = 1e3;
var durationMinute = 6e4;
var durationHour = 36e5;
var durationDay = 864e5;
var durationWeek = 6048e5;
var second = newInterval(function(date) {
date.setTime(Math.floor(date / durationSecond) * durationSecond);
}, function(date, step) {
date.setTime(+date + step * durationSecond);
}, function(start, end) {
return (end - start) / durationSecond;
}, function(date) {
return date.getUTCSeconds();
});
var seconds = second.range;
var minute = newInterval(function(date) {
date.setTime(Math.floor(date / durationMinute) * durationMinute);
}, function(date, step) {
date.setTime(+date + step * durationMinute);
}, function(start, end) {
return (end - start) / durationMinute;
}, function(date) {
return date.getMinutes();
});
var minutes = minute.range;
var hour = newInterval(function(date) {
var offset = date.getTimezoneOffset() * durationMinute % durationHour;
if (offset < 0) offset += durationHour;
date.setTime(Math.floor((+date - offset) / durationHour) * durationHour + offset);
}, function(date, step) {
date.setTime(+date + step * durationHour);
}, function(start, end) {
return (end - start) / durationHour;
}, function(date) {
return date.getHours();
});
var hours = hour.range;
var day = newInterval(function(date) {
date.setHours(0, 0, 0, 0);
}, function(date, step) {
date.setDate(date.getDate() + step);
}, function(start, end) {
return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute) / durationDay;
}, function(date) {
return date.getDate() - 1;
});
var days = day.range;
function weekday(i) {
return newInterval(function(date) {
date.setDate(date.getDate() - (date.getDay() + 7 - i) % 7);
date.setHours(0, 0, 0, 0);
}, function(date, step) {
date.setDate(date.getDate() + step * 7);
}, function(start, end) {
return (end - start - (end.getTimezoneOffset() - start.getTimezoneOffset()) * durationMinute) / durationWeek;
});
}
var timeWeek = weekday(0);
var timeMonday = weekday(1);
var tuesday = weekday(2);
var wednesday = weekday(3);
var thursday = weekday(4);
var friday = weekday(5);
var saturday = weekday(6);
var sundays = timeWeek.range;
var mondays = timeMonday.range;
var tuesdays = tuesday.range;
var wednesdays = wednesday.range;
var thursdays = thursday.range;
var fridays = friday.range;
var saturdays = saturday.range;
var month = newInterval(function(date) {
date.setDate(1);
date.setHours(0, 0, 0, 0);
}, function(date, step) {
date.setMonth(date.getMonth() + step);
}, function(start, end) {
return end.getMonth() - start.getMonth() + (end.getFullYear() - start.getFullYear()) * 12;
}, function(date) {
return date.getMonth();
});
var months = month.range;
var year = newInterval(function(date) {
date.setMonth(0, 1);
date.setHours(0, 0, 0, 0);
}, function(date, step) {
date.setFullYear(date.getFullYear() + step);
}, function(start, end) {
return end.getFullYear() - start.getFullYear();
}, function(date) {
return date.getFullYear();
});
// An optimized implementation for this simple case.
year.every = function(k) {
return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval(function(date) {
date.setFullYear(Math.floor(date.getFullYear() / k) * k);
date.setMonth(0, 1);
date.setHours(0, 0, 0, 0);
}, function(date, step) {
date.setFullYear(date.getFullYear() + step * k);
});
};
var years = year.range;
var utcMinute = newInterval(function(date) {
date.setUTCSeconds(0, 0);
}, function(date, step) {
date.setTime(+date + step * durationMinute);
}, function(start, end) {
return (end - start) / durationMinute;
}, function(date) {
return date.getUTCMinutes();
});
var utcMinutes = utcMinute.range;
var utcHour = newInterval(function(date) {
date.setUTCMinutes(0, 0, 0);
}, function(date, step) {
date.setTime(+date + step * durationHour);
}, function(start, end) {
return (end - start) / durationHour;
}, function(date) {
return date.getUTCHours();
});
var utcHours = utcHour.range;
var utcDay = newInterval(function(date) {
date.setUTCHours(0, 0, 0, 0);
}, function(date, step) {
date.setUTCDate(date.getUTCDate() + step);
}, function(start, end) {
return (end - start) / durationDay;
}, function(date) {
return date.getUTCDate() - 1;
});
var utcDays = utcDay.range;
function utcWeekday(i) {
return newInterval(function(date) {
date.setUTCDate(date.getUTCDate() - (date.getUTCDay() + 7 - i) % 7);
date.setUTCHours(0, 0, 0, 0);
}, function(date, step) {
date.setUTCDate(date.getUTCDate() + step * 7);
}, function(start, end) {
return (end - start) / durationWeek;
});
}
var utcWeek = utcWeekday(0);
var utcMonday = utcWeekday(1);
var utcTuesday = utcWeekday(2);
var utcWednesday = utcWeekday(3);
var utcThursday = utcWeekday(4);
var utcFriday = utcWeekday(5);
var utcSaturday = utcWeekday(6);
var utcSundays = utcWeek.range;
var utcMondays = utcMonday.range;
var utcTuesdays = utcTuesday.range;
var utcWednesdays = utcWednesday.range;
var utcThursdays = utcThursday.range;
var utcFridays = utcFriday.range;
var utcSaturdays = utcSaturday.range;
var utcMonth = newInterval(function(date) {
date.setUTCDate(1);
date.setUTCHours(0, 0, 0, 0);
}, function(date, step) {
date.setUTCMonth(date.getUTCMonth() + step);
}, function(start, end) {
return end.getUTCMonth() - start.getUTCMonth() + (end.getUTCFullYear() - start.getUTCFullYear()) * 12;
}, function(date) {
return date.getUTCMonth();
});
var utcMonths = utcMonth.range;
var utcYear = newInterval(function(date) {
date.setUTCMonth(0, 1);
date.setUTCHours(0, 0, 0, 0);
}, function(date, step) {
date.setUTCFullYear(date.getUTCFullYear() + step);
}, function(start, end) {
return end.getUTCFullYear() - start.getUTCFullYear();
}, function(date) {
return date.getUTCFullYear();
});
// An optimized implementation for this simple case.
utcYear.every = function(k) {
return !isFinite(k = Math.floor(k)) || !(k > 0) ? null : newInterval(function(date) {
date.setUTCFullYear(Math.floor(date.getUTCFullYear() / k) * k);
date.setUTCMonth(0, 1);
date.setUTCHours(0, 0, 0, 0);
}, function(date, step) {
date.setUTCFullYear(date.getUTCFullYear() + step * k);
});
};
var utcYears = utcYear.range;
// Computes the decimal coefficient and exponent of the specified number x with
// significant digits p, where x is positive and p is in [1, 21] or undefined.
// For example, formatDecimal(1.23) returns ["123", 0].
function formatDecimal(x, p) {
if ((i = (x = p ? x.toExponential(p - 1) : x.toExponential()).indexOf("e")) < 0) return null; // NaN, ±Infinity
var i, coefficient = x.slice(0, i);
// The string returned by toExponential either has the form \d\.\d+e[-+]\d+
// (e.g., 1.2e+3) or the form \de[-+]\d+ (e.g., 1e+3).
return [
coefficient.length > 1 ? coefficient[0] + coefficient.slice(2) : coefficient,
+x.slice(i + 1)
];
}
function exponent$1(x) {
return x = formatDecimal(Math.abs(x)), x ? x[1] : NaN;
}
function formatGroup(grouping, thousands) {
return function(value, width) {
var i = value.length,
t = [],
j = 0,
g = grouping[0],
length = 0;
while (i > 0 && g > 0) {
if (length + g + 1 > width) g = Math.max(1, width - length);
t.push(value.substring(i -= g, i + g));
if ((length += g + 1) > width) break;
g = grouping[j = (j + 1) % grouping.length];
}
return t.reverse().join(thousands);
};
}
function formatDefault(x, p) {
x = x.toPrecision(p);
out: for (var n = x.length, i = 1, i0 = -1, i1; i < n; ++i) {
switch (x[i]) {
case ".": i0 = i1 = i; break;
case "0": if (i0 === 0) i0 = i; i1 = i; break;
case "e": break out;
default: if (i0 > 0) i0 = 0; break;
}
}
return i0 > 0 ? x.slice(0, i0) + x.slice(i1 + 1) : x;
}
var prefixExponent;
function formatPrefixAuto(x, p) {
var d = formatDecimal(x, p);
if (!d) return x + "";
var coefficient = d[0],
exponent = d[1],
i = exponent - (prefixExponent = Math.max(-8, Math.min(8, Math.floor(exponent / 3))) * 3) + 1,
n = coefficient.length;
return i === n ? coefficient
: i > n ? coefficient + new Array(i - n + 1).join("0")
: i > 0 ? coefficient.slice(0, i) + "." + coefficient.slice(i)
: "0." + new Array(1 - i).join("0") + formatDecimal(x, Math.max(0, p + i - 1))[0]; // less than 1y!
}
function formatRounded(x, p) {
var d = formatDecimal(x, p);
if (!d) return x + "";
var coefficient = d[0],
exponent = d[1];
return exponent < 0 ? "0." + new Array(-exponent).join("0") + coefficient
: coefficient.length > exponent + 1 ? coefficient.slice(0, exponent + 1) + "." + coefficient.slice(exponent + 1)
: coefficient + new Array(exponent - coefficient.length + 2).join("0");
}
var formatTypes = {
"": formatDefault,
"%": function(x, p) { return (x * 100).toFixed(p); },
"b": function(x) { return Math.round(x).toString(2); },
"c": function(x) { return x + ""; },
"d": function(x) { return Math.round(x).toString(10); },
"e": function(x, p) { return x.toExponential(p); },
"f": function(x, p) { return x.toFixed(p); },
"g": function(x, p) { return x.toPrecision(p); },
"o": function(x) { return Math.round(x).toString(8); },
"p": function(x, p) { return formatRounded(x * 100, p); },
"r": formatRounded,
"s": formatPrefixAuto,
"X": function(x) { return Math.round(x).toString(16).toUpperCase(); },
"x": function(x) { return Math.round(x).toString(16); }
};
// [[fill]align][sign][symbol][0][width][,][.precision][type]
var re = /^(?:(.)?([<>=^]))?([+\-\( ])?([$#])?(0)?(\d+)?(,)?(\.\d+)?([a-z%])?$/i;
function formatSpecifier(specifier) {
return new FormatSpecifier(specifier);
}
function FormatSpecifier(specifier) {
if (!(match = re.exec(specifier))) throw new Error("invalid format: " + specifier);
var match,
fill = match[1] || " ",
align = match[2] || ">",
sign = match[3] || "-",
symbol = match[4] || "",
zero = !!match[5],
width = match[6] && +match[6],
comma = !!match[7],
precision = match[8] && +match[8].slice(1),
type = match[9] || "";
// The "n" type is an alias for ",g".
if (type === "n") comma = true, type = "g";
// Map invalid types to the default format.
else if (!formatTypes[type]) type = "";
// If zero fill is specified, padding goes after sign and before digits.
if (zero || (fill === "0" && align === "=")) zero = true, fill = "0", align = "=";
this.fill = fill;
this.align = align;
this.sign = sign;
this.symbol = symbol;
this.zero = zero;
this.width = width;
this.comma = comma;
this.precision = precision;
this.type = type;
}
FormatSpecifier.prototype.toString = function() {
return this.fill
+ this.align
+ this.sign
+ this.symbol
+ (this.zero ? "0" : "")
+ (this.width == null ? "" : Math.max(1, this.width | 0))
+ (this.comma ? "," : "")
+ (this.precision == null ? "" : "." + Math.max(0, this.precision | 0))
+ this.type;
};
var prefixes = ["y","z","a","f","p","n","\xB5","m","","k","M","G","T","P","E","Z","Y"];
function identity$3(x) {
return x;
}
function formatLocale(locale) {
var group = locale.grouping && locale.thousands ? formatGroup(locale.grouping, locale.thousands) : identity$3,
currency = locale.currency,
decimal = locale.decimal;
function newFormat(specifier) {
specifier = formatSpecifier(specifier);
var fill = specifier.fill,
align = specifier.align,
sign = specifier.sign,
symbol = specifier.symbol,
zero = specifier.zero,
width = specifier.width,
comma = specifier.comma,
precision = specifier.precision,
type = specifier.type;
// Compute the prefix and suffix.
// For SI-prefix, the suffix is lazily computed.
var prefix = symbol === "$" ? currency[0] : symbol === "#" && /[boxX]/.test(type) ? "0" + type.toLowerCase() : "",
suffix = symbol === "$" ? currency[1] : /[%p]/.test(type) ? "%" : "";
// What format function should we use?
// Is this an integer type?
// Can this type generate exponential notation?
var formatType = formatTypes[type],
maybeSuffix = !type || /[defgprs%]/.test(type);
// Set the default precision if not specified,
// or clamp the specified precision to the supported range.
// For significant precision, it must be in [1, 21].
// For fixed precision, it must be in [0, 20].
precision = precision == null ? (type ? 6 : 12)
: /[gprs]/.test(type) ? Math.max(1, Math.min(21, precision))
: Math.max(0, Math.min(20, precision));
function format(value) {
var valuePrefix = prefix,
valueSuffix = suffix,
i, n, c;
if (type === "c") {
valueSuffix = formatType(value) + valueSuffix;
value = "";
} else {
value = +value;
// Convert negative to positive, and compute the prefix.
// Note that -0 is not less than 0, but 1 / -0 is!
var valueNegative = (value < 0 || 1 / value < 0) && (value *= -1, true);
// Perform the initial formatting.
value = formatType(value, precision);
// If the original value was negative, it may be rounded to zero during
// formatting; treat this as (positive) zero.
if (valueNegative) {
i = -1, n = value.length;
valueNegative = false;
while (++i < n) {
if (c = value.charCodeAt(i), (48 < c && c < 58)
|| (type === "x" && 96 < c && c < 103)
|| (type === "X" && 64 < c && c < 71)) {
valueNegative = true;
break;
}
}
}
// Compute the prefix and suffix.
valuePrefix = (valueNegative ? (sign === "(" ? sign : "-") : sign === "-" || sign === "(" ? "" : sign) + valuePrefix;
valueSuffix = valueSuffix + (type === "s" ? prefixes[8 + prefixExponent / 3] : "") + (valueNegative && sign === "(" ? ")" : "");
// Break the formatted value into the integer “value” part that can be
// grouped, and fractional or exponential “suffix” part that is not.
if (maybeSuffix) {
i = -1, n = value.length;
while (++i < n) {
if (c = value.charCodeAt(i), 48 > c || c > 57) {
valueSuffix = (c === 46 ? decimal + value.slice(i + 1) : value.slice(i)) + valueSuffix;
value = value.slice(0, i);
break;
}
}
}
}
// If the fill character is not "0", grouping is applied before padding.
if (comma && !zero) value = group(value, Infinity);
// Compute the padding.
var length = valuePrefix.length + value.length + valueSuffix.length,
padding = length < width ? new Array(width - length + 1).join(fill) : "";
// If the fill character is "0", grouping is applied after padding.
if (comma && zero) value = group(padding + value, padding.length ? width - valueSuffix.length : Infinity), padding = "";
// Reconstruct the final output based on the desired alignment.
switch (align) {
case "<": return valuePrefix + value + valueSuffix + padding;
case "=": return valuePrefix + padding + value + valueSuffix;
case "^": return padding.slice(0, length = padding.length >> 1) + valuePrefix + value + valueSuffix + padding.slice(length);
}
return padding + valuePrefix + value + valueSuffix;
}
format.toString = function() {
return specifier + "";
};
return format;
}
function formatPrefix(specifier, value) {
var f = newFormat((specifier = formatSpecifier(specifier), specifier.type = "f", specifier)),
e = Math.max(-8, Math.min(8, Math.floor(exponent$1(value) / 3))) * 3,
k = Math.pow(10, -e),
prefix = prefixes[8 + e / 3];
return function(value) {
return f(k * value) + prefix;
};
}
return {
format: newFormat,
formatPrefix: formatPrefix
};
}
var locale;
exports.format;
exports.formatPrefix;
defaultLocale({
decimal: ".",
thousands: ",",
grouping: [3],
currency: ["$", ""]
});
function defaultLocale(definition) {
locale = formatLocale(definition);
exports.format = locale.format;
exports.formatPrefix = locale.formatPrefix;
return locale;
}
function precisionFixed(step) {
return Math.max(0, -exponent$1(Math.abs(step)));
}
function precisionPrefix(step, value) {
return Math.max(0, Math.max(-8, Math.min(8, Math.floor(exponent$1(value) / 3))) * 3 - exponent$1(Math.abs(step)));
}
function precisionRound(step, max) {
step = Math.abs(step), max = Math.abs(max) - step;
return Math.max(0, exponent$1(max) - exponent$1(step)) + 1;
}
function localDate(d) {
if (0 <= d.y && d.y < 100) {
var date = new Date(-1, d.m, d.d, d.H, d.M, d.S, d.L);
date.setFullYear(d.y);
return date;
}
return new Date(d.y, d.m, d.d, d.H, d.M, d.S, d.L);
}
function utcDate(d) {
if (0 <= d.y && d.y < 100) {
var date = new Date(Date.UTC(-1, d.m, d.d, d.H, d.M, d.S, d.L));
date.setUTCFullYear(d.y);
return date;
}
return new Date(Date.UTC(d.y, d.m, d.d, d.H, d.M, d.S, d.L));
}
function newYear(y) {
return {y: y, m: 0, d: 1, H: 0, M: 0, S: 0, L: 0};
}
function formatLocale$1(locale) {
var locale_dateTime = locale.dateTime,
locale_date = locale.date,
locale_time = locale.time,
locale_periods = locale.periods,
locale_weekdays = locale.days,
locale_shortWeekdays = locale.shortDays,
locale_months = locale.months,
locale_shortMonths = locale.shortMonths;
var periodRe = formatRe(locale_periods),
periodLookup = formatLookup(locale_periods),
weekdayRe = formatRe(locale_weekdays),
weekdayLookup = formatLookup(locale_weekdays),
shortWeekdayRe = formatRe(locale_shortWeekdays),
shortWeekdayLookup = formatLookup(locale_shortWeekdays),
monthRe = formatRe(locale_months),
monthLookup = formatLookup(locale_months),
shortMonthRe = formatRe(locale_shortMonths),
shortMonthLookup = formatLookup(locale_shortMonths);
var formats = {
"a": formatShortWeekday,
"A": formatWeekday,
"b": formatShortMonth,
"B": formatMonth,
"c": null,
"d": formatDayOfMonth,
"e": formatDayOfMonth,
"H": formatHour24,
"I": formatHour12,
"j": formatDayOfYear,
"L": formatMilliseconds,
"m": formatMonthNumber,
"M": formatMinutes,
"p": formatPeriod,
"S": formatSeconds,
"U": formatWeekNumberSunday,
"w": formatWeekdayNumber,
"W": formatWeekNumberMonday,
"x": null,
"X": null,
"y": formatYear,
"Y": formatFullYear,
"Z": formatZone,
"%": formatLiteralPercent
};
var utcFormats = {
"a": formatUTCShortWeekday,
"A": formatUTCWeekday,
"b": formatUTCShortMonth,
"B": formatUTCMonth,
"c": null,
"d": formatUTCDayOfMonth,
"e": formatUTCDayOfMonth,
"H": formatUTCHour24,
"I": formatUTCHour12,
"j": formatUTCDayOfYear,
"L": formatUTCMilliseconds,
"m": formatUTCMonthNumber,
"M": formatUTCMinutes,
"p": formatUTCPeriod,
"S": formatUTCSeconds,
"U": formatUTCWeekNumberSunday,
"w": formatUTCWeekdayNumber,
"W": formatUTCWeekNumberMonday,
"x": null,
"X": null,
"y": formatUTCYear,
"Y": formatUTCFullYear,
"Z": formatUTCZone,
"%": formatLiteralPercent
};
var parses = {
"a": parseShortWeekday,
"A": parseWeekday,
"b": parseShortMonth,
"B": parseMonth,
"c": parseLocaleDateTime,
"d": parseDayOfMonth,
"e": parseDayOfMonth,
"H": parseHour24,
"I": parseHour24,
"j": parseDayOfYear,
"L": parseMilliseconds,
"m": parseMonthNumber,
"M": parseMinutes,
"p": parsePeriod,
"S": parseSeconds,
"U": parseWeekNumberSunday,
"w": parseWeekdayNumber,
"W": parseWeekNumberMonday,
"x": parseLocaleDate,
"X": parseLocaleTime,
"y": parseYear,
"Y": parseFullYear,
"Z": parseZone,
"%": parseLiteralPercent
};
// These recursive directive definitions must be deferred.
formats.x = newFormat(locale_date, formats);
formats.X = newFormat(locale_time, formats);
formats.c = newFormat(locale_dateTime, formats);
utcFormats.x = newFormat(locale_date, utcFormats);
utcFormats.X = newFormat(locale_time, utcFormats);
utcFormats.c = newFormat(locale_dateTime, utcFormats);
function newFormat(specifier, formats) {
return function(date) {
var string = [],
i = -1,
j = 0,
n = specifier.length,
c,
pad,
format;
if (!(date instanceof Date)) date = new Date(+date);
while (++i < n) {
if (specifier.charCodeAt(i) === 37) {
string.push(specifier.slice(j, i));
if ((pad = pads[c = specifier.charAt(++i)]) != null) c = specifier.charAt(++i);
else pad = c === "e" ? " " : "0";
if (format = formats[c]) c = format(date, pad);
string.push(c);
j = i + 1;
}
}
string.push(specifier.slice(j, i));
return string.join("");
};
}
function newParse(specifier, newDate) {
return function(string) {
var d = newYear(1900),
i = parseSpecifier(d, specifier, string += "", 0);
if (i != string.length) return null;
// The am-pm flag is 0 for AM, and 1 for PM.
if ("p" in d) d.H = d.H % 12 + d.p * 12;
// Convert day-of-week and week-of-year to day-of-year.
if ("W" in d || "U" in d) {
if (!("w" in d)) d.w = "W" in d ? 1 : 0;
var day = "Z" in d ? utcDate(newYear(d.y)).getUTCDay() : newDate(newYear(d.y)).getDay();
d.m = 0;
d.d = "W" in d ? (d.w + 6) % 7 + d.W * 7 - (day + 5) % 7 : d.w + d.U * 7 - (day + 6) % 7;
}
// If a time zone is specified, all fields are interpreted as UTC and then
// offset according to the specified time zone.
if ("Z" in d) {
d.H += d.Z / 100 | 0;
d.M += d.Z % 100;
return utcDate(d);
}
// Otherwise, all fields are in local time.
return newDate(d);
};
}
function parseSpecifier(d, specifier, string, j) {
var i = 0,
n = specifier.length,
m = string.length,
c,
parse;
while (i < n) {
if (j >= m) return -1;
c = specifier.charCodeAt(i++);
if (c === 37) {
c = specifier.charAt(i++);
parse = parses[c in pads ? specifier.charAt(i++) : c];
if (!parse || ((j = parse(d, string, j)) < 0)) return -1;
} else if (c != string.charCodeAt(j++)) {
return -1;
}
}
return j;
}
function parsePeriod(d, string, i) {
var n = periodRe.exec(string.slice(i));
return n ? (d.p = periodLookup[n[0].toLowerCase()], i + n[0].length) : -1;
}
function parseShortWeekday(d, string, i) {
var n = shortWeekdayRe.exec(string.slice(i));
return n ? (d.w = shortWeekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
}
function parseWeekday(d, string, i) {
var n = weekdayRe.exec(string.slice(i));
return n ? (d.w = weekdayLookup[n[0].toLowerCase()], i + n[0].length) : -1;
}
function parseShortMonth(d, string, i) {
var n = shortMonthRe.exec(string.slice(i));
return n ? (d.m = shortMonthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
}
function parseMonth(d, string, i) {
var n = monthRe.exec(string.slice(i));
return n ? (d.m = monthLookup[n[0].toLowerCase()], i + n[0].length) : -1;
}
function parseLocaleDateTime(d, string, i) {
return parseSpecifier(d, locale_dateTime, string, i);
}
function parseLocaleDate(d, string, i) {
return parseSpecifier(d, locale_date, string, i);
}
function parseLocaleTime(d, string, i) {
return parseSpecifier(d, locale_time, string, i);
}
function formatShortWeekday(d) {
return locale_shortWeekdays[d.getDay()];
}
function formatWeekday(d) {
return locale_weekdays[d.getDay()];
}
function formatShortMonth(d) {
return locale_shortMonths[d.getMonth()];
}
function formatMonth(d) {
return locale_months[d.getMonth()];
}
function formatPeriod(d) {
return locale_periods[+(d.getHours() >= 12)];
}
function formatUTCShortWeekday(d) {
return locale_shortWeekdays[d.getUTCDay()];
}
function formatUTCWeekday(d) {
return locale_weekdays[d.getUTCDay()];
}
function formatUTCShortMonth(d) {
return locale_shortMonths[d.getUTCMonth()];
}
function formatUTCMonth(d) {
return locale_months[d.getUTCMonth()];
}
function formatUTCPeriod(d) {
return locale_periods[+(d.getUTCHours() >= 12)];
}
return {
format: function(specifier) {
var f = newFormat(specifier += "", formats);
f.toString = function() { return specifier; };
return f;
},
parse: function(specifier) {
var p = newParse(specifier += "", localDate);
p.toString = function() { return specifier; };
return p;
},
utcFormat: function(specifier) {
var f = newFormat(specifier += "", utcFormats);
f.toString = function() { return specifier; };
return f;
},
utcParse: function(specifier) {
var p = newParse(specifier, utcDate);
p.toString = function() { return specifier; };
return p;
}
};
}
var pads = {"-": "", "_": " ", "0": "0"};
var numberRe = /^\s*\d+/;
var percentRe = /^%/;
var requoteRe = /[\\\^\$\*\+\?\|\[\]\(\)\.\{\}]/g;
function pad(value, fill, width) {
var sign = value < 0 ? "-" : "",
string = (sign ? -value : value) + "",
length = string.length;
return sign + (length < width ? new Array(width - length + 1).join(fill) + string : string);
}
function requote(s) {
return s.replace(requoteRe, "\\$&");
}
function formatRe(names) {
return new RegExp("^(?:" + names.map(requote).join("|") + ")", "i");
}
function formatLookup(names) {
var map = {}, i = -1, n = names.length;
while (++i < n) map[names[i].toLowerCase()] = i;
return map;
}
function parseWeekdayNumber(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 1));
return n ? (d.w = +n[0], i + n[0].length) : -1;
}
function parseWeekNumberSunday(d, string, i) {
var n = numberRe.exec(string.slice(i));
return n ? (d.U = +n[0], i + n[0].length) : -1;
}
function parseWeekNumberMonday(d, string, i) {
var n = numberRe.exec(string.slice(i));
return n ? (d.W = +n[0], i + n[0].length) : -1;
}
function parseFullYear(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 4));
return n ? (d.y = +n[0], i + n[0].length) : -1;
}
function parseYear(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.y = +n[0] + (+n[0] > 68 ? 1900 : 2000), i + n[0].length) : -1;
}
function parseZone(d, string, i) {
var n = /^(Z)|([+-]\d\d)(?:\:?(\d\d))?/.exec(string.slice(i, i + 6));
return n ? (d.Z = n[1] ? 0 : -(n[2] + (n[3] || "00")), i + n[0].length) : -1;
}
function parseMonthNumber(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.m = n[0] - 1, i + n[0].length) : -1;
}
function parseDayOfMonth(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.d = +n[0], i + n[0].length) : -1;
}
function parseDayOfYear(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 3));
return n ? (d.m = 0, d.d = +n[0], i + n[0].length) : -1;
}
function parseHour24(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.H = +n[0], i + n[0].length) : -1;
}
function parseMinutes(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.M = +n[0], i + n[0].length) : -1;
}
function parseSeconds(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 2));
return n ? (d.S = +n[0], i + n[0].length) : -1;
}
function parseMilliseconds(d, string, i) {
var n = numberRe.exec(string.slice(i, i + 3));
return n ? (d.L = +n[0], i + n[0].length) : -1;
}
function parseLiteralPercent(d, string, i) {
var n = percentRe.exec(string.slice(i, i + 1));
return n ? i + n[0].length : -1;
}
function formatDayOfMonth(d, p) {
return pad(d.getDate(), p, 2);
}
function formatHour24(d, p) {
return pad(d.getHours(), p, 2);
}
function formatHour12(d, p) {
return pad(d.getHours() % 12 || 12, p, 2);
}
function formatDayOfYear(d, p) {
return pad(1 + day.count(year(d), d), p, 3);
}
function formatMilliseconds(d, p) {
return pad(d.getMilliseconds(), p, 3);
}
function formatMonthNumber(d, p) {
return pad(d.getMonth() + 1, p, 2);
}
function formatMinutes(d, p) {
return pad(d.getMinutes(), p, 2);
}
function formatSeconds(d, p) {
return pad(d.getSeconds(), p, 2);
}
function formatWeekNumberSunday(d, p) {
return pad(timeWeek.count(year(d), d), p, 2);
}
function formatWeekdayNumber(d) {
return d.getDay();
}
function formatWeekNumberMonday(d, p) {
return pad(timeMonday.count(year(d), d), p, 2);
}
function formatYear(d, p) {
return pad(d.getFullYear() % 100, p, 2);
}
function formatFullYear(d, p) {
return pad(d.getFullYear() % 10000, p, 4);
}
function formatZone(d) {
var z = d.getTimezoneOffset();
return (z > 0 ? "-" : (z *= -1, "+"))
+ pad(z / 60 | 0, "0", 2)
+ pad(z % 60, "0", 2);
}
function formatUTCDayOfMonth(d, p) {
return pad(d.getUTCDate(), p, 2);
}
function formatUTCHour24(d, p) {
return pad(d.getUTCHours(), p, 2);
}
function formatUTCHour12(d, p) {
return pad(d.getUTCHours() % 12 || 12, p, 2);
}
function formatUTCDayOfYear(d, p) {
return pad(1 + utcDay.count(utcYear(d), d), p, 3);
}
function formatUTCMilliseconds(d, p) {
return pad(d.getUTCMilliseconds(), p, 3);
}
function formatUTCMonthNumber(d, p) {
return pad(d.getUTCMonth() + 1, p, 2);
}
function formatUTCMinutes(d, p) {
return pad(d.getUTCMinutes(), p, 2);
}
function formatUTCSeconds(d, p) {
return pad(d.getUTCSeconds(), p, 2);
}
function formatUTCWeekNumberSunday(d, p) {
return pad(utcWeek.count(utcYear(d), d), p, 2);
}
function formatUTCWeekdayNumber(d) {
return d.getUTCDay();
}
function formatUTCWeekNumberMonday(d, p) {
return pad(utcMonday.count(utcYear(d), d), p, 2);
}
function formatUTCYear(d, p) {
return pad(d.getUTCFullYear() % 100, p, 2);
}
function formatUTCFullYear(d, p) {
return pad(d.getUTCFullYear() % 10000, p, 4);
}
function formatUTCZone() {
return "+0000";
}
function formatLiteralPercent() {
return "%";
}
var locale$1;
exports.timeFormat;
exports.timeParse;
exports.utcFormat;
exports.utcParse;
defaultLocale$1({
dateTime: "%x, %X",
date: "%-m/%-d/%Y",
time: "%-I:%M:%S %p",
periods: ["AM", "PM"],
days: ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"],
shortDays: ["Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"],
months: ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"],
shortMonths: ["Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
});
function defaultLocale$1(definition) {
locale$1 = formatLocale$1(definition);
exports.timeFormat = locale$1.format;
exports.timeParse = locale$1.parse;
exports.utcFormat = locale$1.utcFormat;
exports.utcParse = locale$1.utcParse;
return locale$1;
}
var isoSpecifier = "%Y-%m-%dT%H:%M:%S.%LZ";
function formatIsoNative(date) {
return date.toISOString();
}
var formatIso = Date.prototype.toISOString
? formatIsoNative
: exports.utcFormat(isoSpecifier);
function parseIsoNative(string) {
var date = new Date(string);
return isNaN(date) ? null : date;
}
var parseIso = +new Date("2000-01-01T00:00:00.000Z")
? parseIsoNative
: exports.utcParse(isoSpecifier);
var array$2 = Array.prototype;
var map$2 = array$2.map;
var slice$3 = array$2.slice;
var implicit = {name: "implicit"};
function ordinal(range) {
var index = map$1(),
domain = [],
unknown = implicit;
range = range == null ? [] : slice$3.call(range);
function scale(d) {
var key = d + "", i = index.get(key);
if (!i) {
if (unknown !== implicit) return unknown;
index.set(key, i = domain.push(d));
}
return range[(i - 1) % range.length];
}
scale.domain = function(_) {
if (!arguments.length) return domain.slice();
domain = [], index = map$1();
var i = -1, n = _.length, d, key;
while (++i < n) if (!index.has(key = (d = _[i]) + "")) index.set(key, domain.push(d));
return scale;
};
scale.range = function(_) {
return arguments.length ? (range = slice$3.call(_), scale) : range.slice();
};
scale.unknown = function(_) {
return arguments.length ? (unknown = _, scale) : unknown;
};
scale.copy = function() {
return ordinal()
.domain(domain)
.range(range)
.unknown(unknown);
};
return scale;
}
function band() {
var scale = ordinal().unknown(undefined),
domain = scale.domain,
ordinalRange = scale.range,
range$$ = [0, 1],
step,
bandwidth,
round = false,
paddingInner = 0,
paddingOuter = 0,
align = 0.5;
delete scale.unknown;
function rescale() {
var n = domain().length,
reverse = range$$[1] < range$$[0],
start = range$$[reverse - 0],
stop = range$$[1 - reverse];
step = (stop - start) / Math.max(1, n - paddingInner + paddingOuter * 2);
if (round) step = Math.floor(step);
start += (stop - start - step * (n - paddingInner)) * align;
bandwidth = step * (1 - paddingInner);
if (round) start = Math.round(start), bandwidth = Math.round(bandwidth);
var values = range(n).map(function(i) { return start + step * i; });
return ordinalRange(reverse ? values.reverse() : values);
}
scale.domain = function(_) {
return arguments.length ? (domain(_), rescale()) : domain();
};
scale.range = function(_) {
return arguments.length ? (range$$ = [+_[0], +_[1]], rescale()) : range$$.slice();
};
scale.rangeRound = function(_) {
return range$$ = [+_[0], +_[1]], round = true, rescale();
};
scale.bandwidth = function() {
return bandwidth;
};
scale.step = function() {
return step;
};
scale.round = function(_) {
return arguments.length ? (round = !!_, rescale()) : round;
};
scale.padding = function(_) {
return arguments.length ? (paddingInner = paddingOuter = Math.max(0, Math.min(1, _)), rescale()) : paddingInner;
};
scale.paddingInner = function(_) {
return arguments.length ? (paddingInner = Math.max(0, Math.min(1, _)), rescale()) : paddingInner;
};
scale.paddingOuter = function(_) {
return arguments.length ? (paddingOuter = Math.max(0, Math.min(1, _)), rescale()) : paddingOuter;
};
scale.align = function(_) {
return arguments.length ? (align = Math.max(0, Math.min(1, _)), rescale()) : align;
};
scale.copy = function() {
return band()
.domain(domain())
.range(range$$)
.round(round)
.paddingInner(paddingInner)
.paddingOuter(paddingOuter)
.align(align);
};
return rescale();
}
function pointish(scale) {
var copy = scale.copy;
scale.padding = scale.paddingOuter;
delete scale.paddingInner;
delete scale.paddingOuter;
scale.copy = function() {
return pointish(copy());
};
return scale;
}
function point$4() {
return pointish(band().paddingInner(1));
}
function constant$3(x) {
return function() {
return x;
};
}
function number$1(x) {
return +x;
}
var unit = [0, 1];
function deinterpolate(a, b) {
return (b -= (a = +a))
? function(x) { return (x - a) / b; }
: constant$3(b);
}
function deinterpolateClamp(deinterpolate) {
return function(a, b) {
var d = deinterpolate(a = +a, b = +b);
return function(x) { return x <= a ? 0 : x >= b ? 1 : d(x); };
};
}
function reinterpolateClamp(reinterpolate) {
return function(a, b) {
var r = reinterpolate(a = +a, b = +b);
return function(t) { return t <= 0 ? a : t >= 1 ? b : r(t); };
};
}
function bimap(domain, range, deinterpolate, reinterpolate) {
var d0 = domain[0], d1 = domain[1], r0 = range[0], r1 = range[1];
if (d1 < d0) d0 = deinterpolate(d1, d0), r0 = reinterpolate(r1, r0);
else d0 = deinterpolate(d0, d1), r0 = reinterpolate(r0, r1);
return function(x) { return r0(d0(x)); };
}
function polymap(domain, range, deinterpolate, reinterpolate) {
var j = Math.min(domain.length, range.length) - 1,
d = new Array(j),
r = new Array(j),
i = -1;
// Reverse descending domains.
if (domain[j] < domain[0]) {
domain = domain.slice().reverse();
range = range.slice().reverse();
}
while (++i < j) {
d[i] = deinterpolate(domain[i], domain[i + 1]);
r[i] = reinterpolate(range[i], range[i + 1]);
}
return function(x) {
var i = bisectRight(domain, x, 1, j) - 1;
return r[i](d[i](x));
};
}
function copy(source, target) {
return target
.domain(source.domain())
.range(source.range())
.interpolate(source.interpolate())
.clamp(source.clamp());
}
// deinterpolate(a, b)(x) takes a domain value x in [a,b] and returns the corresponding parameter t in [0,1].
// reinterpolate(a, b)(t) takes a parameter t in [0,1] and returns the corresponding domain value x in [a,b].
function continuous(deinterpolate$$, reinterpolate) {
var domain = unit,
range = unit,
interpolate$$ = interpolate,
clamp = false,
piecewise,
output,
input;
function rescale() {
piecewise = Math.min(domain.length, range.length) > 2 ? polymap : bimap;
output = input = null;
return scale;
}
function scale(x) {
return (output || (output = piecewise(domain, range, clamp ? deinterpolateClamp(deinterpolate$$) : deinterpolate$$, interpolate$$)))(+x);
}
scale.invert = function(y) {
return (input || (input = piecewise(range, domain, deinterpolate, clamp ? reinterpolateClamp(reinterpolate) : reinterpolate)))(+y);
};
scale.domain = function(_) {
return arguments.length ? (domain = map$2.call(_, number$1), rescale()) : domain.slice();
};
scale.range = function(_) {
return arguments.length ? (range = slice$3.call(_), rescale()) : range.slice();
};
scale.rangeRound = function(_) {
return range = slice$3.call(_), interpolate$$ = interpolateRound, rescale();
};
scale.clamp = function(_) {
return arguments.length ? (clamp = !!_, rescale()) : clamp;
};
scale.interpolate = function(_) {
return arguments.length ? (interpolate$$ = _, rescale()) : interpolate$$;
};
return rescale();
}
function tickFormat(domain, count, specifier) {
var start = domain[0],
stop = domain[domain.length - 1],
step = tickStep(start, stop, count == null ? 10 : count),
precision;
specifier = formatSpecifier(specifier == null ? ",f" : specifier);
switch (specifier.type) {
case "s": {
var value = Math.max(Math.abs(start), Math.abs(stop));
if (specifier.precision == null && !isNaN(precision = precisionPrefix(step, value))) specifier.precision = precision;
return exports.formatPrefix(specifier, value);
}
case "":
case "e":
case "g":
case "p":
case "r": {
if (specifier.precision == null && !isNaN(precision = precisionRound(step, Math.max(Math.abs(start), Math.abs(stop))))) specifier.precision = precision - (specifier.type === "e");
break;
}
case "f":
case "%": {
if (specifier.precision == null && !isNaN(precision = precisionFixed(step))) specifier.precision = precision - (specifier.type === "%") * 2;
break;
}
}
return exports.format(specifier);
}
function linearish(scale) {
var domain = scale.domain;
scale.ticks = function(count) {
var d = domain();
return ticks(d[0], d[d.length - 1], count == null ? 10 : count);
};
scale.tickFormat = function(count, specifier) {
return tickFormat(domain(), count, specifier);
};
scale.nice = function(count) {
var d = domain(),
i = d.length - 1,
n = count == null ? 10 : count,
start = d[0],
stop = d[i],
step = tickStep(start, stop, n);
if (step) {
step = tickStep(Math.floor(start / step) * step, Math.ceil(stop / step) * step, n);
d[0] = Math.floor(start / step) * step;
d[i] = Math.ceil(stop / step) * step;
domain(d);
}
return scale;
};
return scale;
}
function linear$2() {
var scale = continuous(deinterpolate, interpolateNumber);
scale.copy = function() {
return copy(scale, linear$2());
};
return linearish(scale);
}
function identity$4() {
var domain = [0, 1];
function scale(x) {
return +x;
}
scale.invert = scale;
scale.domain = scale.range = function(_) {
return arguments.length ? (domain = map$2.call(_, number$1), scale) : domain.slice();
};
scale.copy = function() {
return identity$4().domain(domain);
};
return linearish(scale);
}
function nice(domain, interval) {
domain = domain.slice();
var i0 = 0,
i1 = domain.length - 1,
x0 = domain[i0],
x1 = domain[i1],
t;
if (x1 < x0) {
t = i0, i0 = i1, i1 = t;
t = x0, x0 = x1, x1 = t;
}
domain[i0] = interval.floor(x0);
domain[i1] = interval.ceil(x1);
return domain;
}
function deinterpolate$1(a, b) {
return (b = Math.log(b / a))
? function(x) { return Math.log(x / a) / b; }
: constant$3(b);
}
function reinterpolate(a, b) {
return a < 0
? function(t) { return -Math.pow(-b, t) * Math.pow(-a, 1 - t); }
: function(t) { return Math.pow(b, t) * Math.pow(a, 1 - t); };
}
function pow10(x) {
return isFinite(x) ? +("1e" + x) : x < 0 ? 0 : x;
}
function powp(base) {
return base === 10 ? pow10
: base === Math.E ? Math.exp
: function(x) { return Math.pow(base, x); };
}
function logp(base) {
return base === Math.E ? Math.log
: base === 10 && Math.log10
|| base === 2 && Math.log2
|| (base = Math.log(base), function(x) { return Math.log(x) / base; });
}
function reflect(f) {
return function(x) {
return -f(-x);
};
}
function log() {
var scale = continuous(deinterpolate$1, reinterpolate).domain([1, 10]),
domain = scale.domain,
base = 10,
logs = logp(10),
pows = powp(10);
function rescale() {
logs = logp(base), pows = powp(base);
if (domain()[0] < 0) logs = reflect(logs), pows = reflect(pows);
return scale;
}
scale.base = function(_) {
return arguments.length ? (base = +_, rescale()) : base;
};
scale.domain = function(_) {
return arguments.length ? (domain(_), rescale()) : domain();
};
scale.ticks = function(count) {
var d = domain(),
u = d[0],
v = d[d.length - 1],
r;
if (r = v < u) i = u, u = v, v = i;
var i = logs(u),
j = logs(v),
p,
k,
t,
n = count == null ? 10 : +count,
z = [];
if (!(base % 1) && j - i < n) {
i = Math.round(i) - 1, j = Math.round(j) + 1;
if (u > 0) for (; i < j; ++i) {
for (k = 1, p = pows(i); k < base; ++k) {
t = p * k;
if (t < u) continue;
if (t > v) break;
z.push(t);
}
} else for (; i < j; ++i) {
for (k = base - 1, p = pows(i); k >= 1; --k) {
t = p * k;
if (t < u) continue;
if (t > v) break;
z.push(t);
}
}
} else {
z = ticks(i, j, Math.min(j - i, n)).map(pows);
}
return r ? z.reverse() : z;
};
scale.tickFormat = function(count, specifier) {
if (specifier == null) specifier = base === 10 ? ".0e" : ",";
if (typeof specifier !== "function") specifier = exports.format(specifier);
if (count === Infinity) return specifier;
if (count == null) count = 10;
var k = Math.max(1, base * count / scale.ticks().length); // TODO fast estimate?
return function(d) {
var i = d / pows(Math.round(logs(d)));
if (i * base < base - 0.5) i *= base;
return i <= k ? specifier(d) : "";
};
};
scale.nice = function() {
return domain(nice(domain(), {
floor: function(x) { return pows(Math.floor(logs(x))); },
ceil: function(x) { return pows(Math.ceil(logs(x))); }
}));
};
scale.copy = function() {
return copy(scale, log().base(base));
};
return scale;
}
function raise(x, exponent) {
return x < 0 ? -Math.pow(-x, exponent) : Math.pow(x, exponent);
}
function pow() {
var exponent = 1,
scale = continuous(deinterpolate, reinterpolate),
domain = scale.domain;
function deinterpolate(a, b) {
return (b = raise(b, exponent) - (a = raise(a, exponent)))
? function(x) { return (raise(x, exponent) - a) / b; }
: constant$3(b);
}
function reinterpolate(a, b) {
b = raise(b, exponent) - (a = raise(a, exponent));
return function(t) { return raise(a + b * t, 1 / exponent); };
}
scale.exponent = function(_) {
return arguments.length ? (exponent = +_, domain(domain())) : exponent;
};
scale.copy = function() {
return copy(scale, pow().exponent(exponent));
};
return linearish(scale);
}
function sqrt() {
return pow().exponent(0.5);
}
function quantile() {
var domain = [],
range = [],
thresholds = [];
function rescale() {
var i = 0, n = Math.max(1, range.length);
thresholds = new Array(n - 1);
while (++i < n) thresholds[i - 1] = threshold(domain, i / n);
return scale;
}
function scale(x) {
if (!isNaN(x = +x)) return range[bisectRight(thresholds, x)];
}
scale.invertExtent = function(y) {
var i = range.indexOf(y);
return i < 0 ? [NaN, NaN] : [
i > 0 ? thresholds[i - 1] : domain[0],
i < thresholds.length ? thresholds[i] : domain[domain.length - 1]
];
};
scale.domain = function(_) {
if (!arguments.length) return domain.slice();
domain = [];
for (var i = 0, n = _.length, d; i < n; ++i) if (d = _[i], d != null && !isNaN(d = +d)) domain.push(d);
domain.sort(ascending);
return rescale();
};
scale.range = function(_) {
return arguments.length ? (range = slice$3.call(_), rescale()) : range.slice();
};
scale.quantiles = function() {
return thresholds.slice();
};
scale.copy = function() {
return quantile()
.domain(domain)
.range(range);
};
return scale;
}
function quantize$1() {
var x0 = 0,
x1 = 1,
n = 1,
domain = [0.5],
range = [0, 1];
function scale(x) {
if (x <= x) return range[bisectRight(domain, x, 0, n)];
}
function rescale() {
var i = -1;
domain = new Array(n);
while (++i < n) domain[i] = ((i + 1) * x1 - (i - n) * x0) / (n + 1);
return scale;
}
scale.domain = function(_) {
return arguments.length ? (x0 = +_[0], x1 = +_[1], rescale()) : [x0, x1];
};
scale.range = function(_) {
return arguments.length ? (n = (range = slice$3.call(_)).length - 1, rescale()) : range.slice();
};
scale.invertExtent = function(y) {
var i = range.indexOf(y);
return i < 0 ? [NaN, NaN]
: i < 1 ? [x0, domain[0]]
: i >= n ? [domain[n - 1], x1]
: [domain[i - 1], domain[i]];
};
scale.copy = function() {
return quantize$1()
.domain([x0, x1])
.range(range);
};
return linearish(scale);
}
function threshold$1() {
var domain = [0.5],
range = [0, 1],
n = 1;
function scale(x) {
if (x <= x) return range[bisectRight(domain, x, 0, n)];
}
scale.domain = function(_) {
return arguments.length ? (domain = slice$3.call(_), n = Math.min(domain.length, range.length - 1), scale) : domain.slice();
};
scale.range = function(_) {
return arguments.length ? (range = slice$3.call(_), n = Math.min(domain.length, range.length - 1), scale) : range.slice();
};
scale.invertExtent = function(y) {
var i = range.indexOf(y);
return [domain[i - 1], domain[i]];
};
scale.copy = function() {
return threshold$1()
.domain(domain)
.range(range);
};
return scale;
}
var durationSecond$1 = 1000;
var durationMinute$1 = durationSecond$1 * 60;
var durationHour$1 = durationMinute$1 * 60;
var durationDay$1 = durationHour$1 * 24;
var durationWeek$1 = durationDay$1 * 7;
var durationMonth = durationDay$1 * 30;
var durationYear = durationDay$1 * 365;
function date$1(t) {
return new Date(t);
}
function number$2(t) {
return t instanceof Date ? +t : +new Date(+t);
}
function calendar(year, month, week, day, hour, minute, second, millisecond, format) {
var scale = continuous(deinterpolate, interpolateNumber),
invert = scale.invert,
domain = scale.domain;
var formatMillisecond = format(".%L"),
formatSecond = format(":%S"),
formatMinute = format("%I:%M"),
formatHour = format("%I %p"),
formatDay = format("%a %d"),
formatWeek = format("%b %d"),
formatMonth = format("%B"),
formatYear = format("%Y");
var tickIntervals = [
[second, 1, durationSecond$1],
[second, 5, 5 * durationSecond$1],
[second, 15, 15 * durationSecond$1],
[second, 30, 30 * durationSecond$1],
[minute, 1, durationMinute$1],
[minute, 5, 5 * durationMinute$1],
[minute, 15, 15 * durationMinute$1],
[minute, 30, 30 * durationMinute$1],
[ hour, 1, durationHour$1 ],
[ hour, 3, 3 * durationHour$1 ],
[ hour, 6, 6 * durationHour$1 ],
[ hour, 12, 12 * durationHour$1 ],
[ day, 1, durationDay$1 ],
[ day, 2, 2 * durationDay$1 ],
[ week, 1, durationWeek$1 ],
[ month, 1, durationMonth ],
[ month, 3, 3 * durationMonth ],
[ year, 1, durationYear ]
];
function tickFormat(date) {
return (second(date) < date ? formatMillisecond
: minute(date) < date ? formatSecond
: hour(date) < date ? formatMinute
: day(date) < date ? formatHour
: month(date) < date ? (week(date) < date ? formatDay : formatWeek)
: year(date) < date ? formatMonth
: formatYear)(date);
}
function tickInterval(interval, start, stop, step) {
if (interval == null) interval = 10;
// If a desired tick count is specified, pick a reasonable tick interval
// based on the extent of the domain and a rough estimate of tick size.
// Otherwise, assume interval is already a time interval and use it.
if (typeof interval === "number") {
var target = Math.abs(stop - start) / interval,
i = bisector(function(i) { return i[2]; }).right(tickIntervals, target);
if (i === tickIntervals.length) {
step = tickStep(start / durationYear, stop / durationYear, interval);
interval = year;
} else if (i) {
i = tickIntervals[target / tickIntervals[i - 1][2] < tickIntervals[i][2] / target ? i - 1 : i];
step = i[1];
interval = i[0];
} else {
step = tickStep(start, stop, interval);
interval = millisecond;
}
}
return step == null ? interval : interval.every(step);
}
scale.invert = function(y) {
return new Date(invert(y));
};
scale.domain = function(_) {
return arguments.length ? domain(map$2.call(_, number$2)) : domain().map(date$1);
};
scale.ticks = function(interval, step) {
var d = domain(),
t0 = d[0],
t1 = d[d.length - 1],
r = t1 < t0,
t;
if (r) t = t0, t0 = t1, t1 = t;
t = tickInterval(interval, t0, t1, step);
t = t ? t.range(t0, t1 + 1) : []; // inclusive stop
return r ? t.reverse() : t;
};
scale.tickFormat = function(count, specifier) {
return specifier == null ? tickFormat : format(specifier);
};
scale.nice = function(interval, step) {
var d = domain();
return (interval = tickInterval(interval, d[0], d[d.length - 1], step))
? domain(nice(d, interval))
: scale;
};
scale.copy = function() {
return copy(scale, calendar(year, month, week, day, hour, minute, second, millisecond, format));
};
return scale;
}
function time() {
return calendar(year, month, timeWeek, day, hour, minute, second, millisecond, exports.timeFormat).domain([new Date(2000, 0, 1), new Date(2000, 0, 2)]);
}
function utcTime() {
return calendar(utcYear, utcMonth, utcWeek, utcDay, utcHour, utcMinute, second, millisecond, exports.utcFormat).domain([Date.UTC(2000, 0, 1), Date.UTC(2000, 0, 2)]);
}
function colors(s) {
return s.match(/.{6}/g).map(function(x) {
return "#" + x;
});
}
var category10 = colors("1f77b4ff7f0e2ca02cd627289467bd8c564be377c27f7f7fbcbd2217becf");
var category20b = colors("393b795254a36b6ecf9c9ede6379398ca252b5cf6bcedb9c8c6d31bd9e39e7ba52e7cb94843c39ad494ad6616be7969c7b4173a55194ce6dbdde9ed6");
var category20c = colors("3182bd6baed69ecae1c6dbefe6550dfd8d3cfdae6bfdd0a231a35474c476a1d99bc7e9c0756bb19e9ac8bcbddcdadaeb636363969696bdbdbdd9d9d9");
var category20 = colors("1f77b4aec7e8ff7f0effbb782ca02c98df8ad62728ff98969467bdc5b0d58c564bc49c94e377c2f7b6d27f7f7fc7c7c7bcbd22dbdb8d17becf9edae5");
var cubehelix$3 = interpolateCubehelixLong(cubehelix(300, 0.5, 0.0), cubehelix(-240, 0.5, 1.0));
var warm = interpolateCubehelixLong(cubehelix(-100, 0.75, 0.35), cubehelix(80, 1.50, 0.8));
var cool = interpolateCubehelixLong(cubehelix(260, 0.75, 0.35), cubehelix(80, 1.50, 0.8));
var rainbow = cubehelix();
function rainbow$1(t) {
if (t < 0 || t > 1) t -= Math.floor(t);
var ts = Math.abs(t - 0.5);
rainbow.h = 360 * t - 100;
rainbow.s = 1.5 - 1.5 * ts;
rainbow.l = 0.8 - 0.9 * ts;
return rainbow + "";
}
function ramp(range) {
var n = range.length;
return function(t) {
return range[Math.max(0, Math.min(n - 1, Math.floor(t * n)))];
};
}
var viridis = ramp(colors("44015444025645045745055946075a46085c460a5d460b5e470d60470e6147106347116447136548146748166848176948186a481a6c481b6d481c6e481d6f481f70482071482173482374482475482576482677482878482979472a7a472c7a472d7b472e7c472f7d46307e46327e46337f463480453581453781453882443983443a83443b84433d84433e85423f854240864241864142874144874045884046883f47883f48893e49893e4a893e4c8a3d4d8a3d4e8a3c4f8a3c508b3b518b3b528b3a538b3a548c39558c39568c38588c38598c375a8c375b8d365c8d365d8d355e8d355f8d34608d34618d33628d33638d32648e32658e31668e31678e31688e30698e306a8e2f6b8e2f6c8e2e6d8e2e6e8e2e6f8e2d708e2d718e2c718e2c728e2c738e2b748e2b758e2a768e2a778e2a788e29798e297a8e297b8e287c8e287d8e277e8e277f8e27808e26818e26828e26828e25838e25848e25858e24868e24878e23888e23898e238a8d228b8d228c8d228d8d218e8d218f8d21908d21918c20928c20928c20938c1f948c1f958b1f968b1f978b1f988b1f998a1f9a8a1e9b8a1e9c891e9d891f9e891f9f881fa0881fa1881fa1871fa28720a38620a48621a58521a68522a78522a88423a98324aa8325ab8225ac8226ad8127ad8128ae8029af7f2ab07f2cb17e2db27d2eb37c2fb47c31b57b32b67a34b67935b77937b87838b9773aba763bbb753dbc743fbc7340bd7242be7144bf7046c06f48c16e4ac16d4cc26c4ec36b50c46a52c56954c56856c66758c7655ac8645cc8635ec96260ca6063cb5f65cb5e67cc5c69cd5b6ccd5a6ece5870cf5773d05675d05477d1537ad1517cd2507fd34e81d34d84d44b86d54989d5488bd6468ed64590d74393d74195d84098d83e9bd93c9dd93ba0da39a2da37a5db36a8db34aadc32addc30b0dd2fb2dd2db5de2bb8de29bade28bddf26c0df25c2df23c5e021c8e020cae11fcde11dd0e11cd2e21bd5e21ad8e219dae319dde318dfe318e2e418e5e419e7e419eae51aece51befe51cf1e51df4e61ef6e620f8e621fbe723fde725"));
var magma = ramp(colors("00000401000501010601010802010902020b02020d03030f03031204041405041606051806051a07061c08071e0907200a08220b09240c09260d0a290e0b2b100b2d110c2f120d31130d34140e36150e38160f3b180f3d19103f1a10421c10441d11471e114920114b21114e22115024125325125527125829115a2a115c2c115f2d11612f116331116533106734106936106b38106c390f6e3b0f703d0f713f0f72400f74420f75440f764510774710784910784a10794c117a4e117b4f127b51127c52137c54137d56147d57157e59157e5a167e5c167f5d177f5f187f601880621980641a80651a80671b80681c816a1c816b1d816d1d816e1e81701f81721f817320817521817621817822817922827b23827c23827e24828025828125818326818426818627818827818928818b29818c29818e2a81902a81912b81932b80942c80962c80982d80992d809b2e7f9c2e7f9e2f7fa02f7fa1307ea3307ea5317ea6317da8327daa337dab337cad347cae347bb0357bb2357bb3367ab5367ab73779b83779ba3878bc3978bd3977bf3a77c03a76c23b75c43c75c53c74c73d73c83e73ca3e72cc3f71cd4071cf4070d0416fd2426fd3436ed5446dd6456cd8456cd9466bdb476adc4869de4968df4a68e04c67e24d66e34e65e44f64e55064e75263e85362e95462ea5661eb5760ec5860ed5a5fee5b5eef5d5ef05f5ef1605df2625df2645cf3655cf4675cf4695cf56b5cf66c5cf66e5cf7705cf7725cf8745cf8765cf9785df9795df97b5dfa7d5efa7f5efa815ffb835ffb8560fb8761fc8961fc8a62fc8c63fc8e64fc9065fd9266fd9467fd9668fd9869fd9a6afd9b6bfe9d6cfe9f6dfea16efea36ffea571fea772fea973feaa74feac76feae77feb078feb27afeb47bfeb67cfeb77efeb97ffebb81febd82febf84fec185fec287fec488fec68afec88cfeca8dfecc8ffecd90fecf92fed194fed395fed597fed799fed89afdda9cfddc9efddea0fde0a1fde2a3fde3a5fde5a7fde7a9fde9aafdebacfcecaefceeb0fcf0b2fcf2b4fcf4b6fcf6b8fcf7b9fcf9bbfcfbbdfcfdbf"));
var inferno = ramp(colors("00000401000501010601010802010a02020c02020e03021004031204031405041706041907051b08051d09061f0a07220b07240c08260d08290e092b10092d110a30120a32140b34150b37160b39180c3c190c3e1b0c411c0c431e0c451f0c48210c4a230c4c240c4f260c51280b53290b552b0b572d0b592f0a5b310a5c320a5e340a5f3609613809623909633b09643d09653e0966400a67420a68440a68450a69470b6a490b6a4a0c6b4c0c6b4d0d6c4f0d6c510e6c520e6d540f6d550f6d57106e59106e5a116e5c126e5d126e5f136e61136e62146e64156e65156e67166e69166e6a176e6c186e6d186e6f196e71196e721a6e741a6e751b6e771c6d781c6d7a1d6d7c1d6d7d1e6d7f1e6c801f6c82206c84206b85216b87216b88226a8a226a8c23698d23698f24699025689225689326679526679727669827669a28659b29649d29649f2a63a02a63a22b62a32c61a52c60a62d60a82e5fa92e5eab2f5ead305dae305cb0315bb1325ab3325ab43359b63458b73557b93556ba3655bc3754bd3853bf3952c03a51c13a50c33b4fc43c4ec63d4dc73e4cc83f4bca404acb4149cc4248ce4347cf4446d04545d24644d34743d44842d54a41d74b3fd84c3ed94d3dda4e3cdb503bdd513ade5238df5337e05536e15635e25734e35933e45a31e55c30e65d2fe75e2ee8602de9612bea632aeb6429eb6628ec6726ed6925ee6a24ef6c23ef6e21f06f20f1711ff1731df2741cf3761bf37819f47918f57b17f57d15f67e14f68013f78212f78410f8850ff8870ef8890cf98b0bf98c0af98e09fa9008fa9207fa9407fb9606fb9706fb9906fb9b06fb9d07fc9f07fca108fca309fca50afca60cfca80dfcaa0ffcac11fcae12fcb014fcb216fcb418fbb61afbb81dfbba1ffbbc21fbbe23fac026fac228fac42afac62df9c72ff9c932f9cb35f8cd37f8cf3af7d13df7d340f6d543f6d746f5d949f5db4cf4dd4ff4df53f4e156f3e35af3e55df2e661f2e865f2ea69f1ec6df1ed71f1ef75f1f179f2f27df2f482f3f586f3f68af4f88ef5f992f6fa96f8fb9af9fc9dfafda1fcffa4"));
var plasma = ramp(colors("0d088710078813078916078a19068c1b068d1d068e20068f2206902406912605912805922a05932c05942e05952f059631059733059735049837049938049a3a049a3c049b3e049c3f049c41049d43039e44039e46039f48039f4903a04b03a14c02a14e02a25002a25102a35302a35502a45601a45801a45901a55b01a55c01a65e01a66001a66100a76300a76400a76600a76700a86900a86a00a86c00a86e00a86f00a87100a87201a87401a87501a87701a87801a87a02a87b02a87d03a87e03a88004a88104a78305a78405a78606a68707a68808a68a09a58b0aa58d0ba58e0ca48f0da4910ea3920fa39410a29511a19613a19814a099159f9a169f9c179e9d189d9e199da01a9ca11b9ba21d9aa31e9aa51f99a62098a72197a82296aa2395ab2494ac2694ad2793ae2892b02991b12a90b22b8fb32c8eb42e8db52f8cb6308bb7318ab83289ba3388bb3488bc3587bd3786be3885bf3984c03a83c13b82c23c81c33d80c43e7fc5407ec6417dc7427cc8437bc9447aca457acb4679cc4778cc4977cd4a76ce4b75cf4c74d04d73d14e72d24f71d35171d45270d5536fd5546ed6556dd7566cd8576bd9586ada5a6ada5b69db5c68dc5d67dd5e66de5f65de6164df6263e06363e16462e26561e26660e3685fe4695ee56a5de56b5de66c5ce76e5be76f5ae87059e97158e97257ea7457eb7556eb7655ec7754ed7953ed7a52ee7b51ef7c51ef7e50f07f4ff0804ef1814df1834cf2844bf3854bf3874af48849f48948f58b47f58c46f68d45f68f44f79044f79143f79342f89441f89540f9973ff9983ef99a3efa9b3dfa9c3cfa9e3bfb9f3afba139fba238fca338fca537fca636fca835fca934fdab33fdac33fdae32fdaf31fdb130fdb22ffdb42ffdb52efeb72dfeb82cfeba2cfebb2bfebd2afebe2afec029fdc229fdc328fdc527fdc627fdc827fdca26fdcb26fccd25fcce25fcd025fcd225fbd324fbd524fbd724fad824fada24f9dc24f9dd25f8df25f8e125f7e225f7e425f6e626f6e826f5e926f5eb27f4ed27f3ee27f3f027f2f227f1f426f1f525f0f724f0f921"));
function sequential(interpolator) {
var x0 = 0,
x1 = 1,
clamp = false;
function scale(x) {
var t = (x - x0) / (x1 - x0);
return interpolator(clamp ? Math.max(0, Math.min(1, t)) : t);
}
scale.domain = function(_) {
return arguments.length ? (x0 = +_[0], x1 = +_[1], scale) : [x0, x1];
};
scale.clamp = function(_) {
return arguments.length ? (clamp = !!_, scale) : clamp;
};
scale.interpolator = function(_) {
return arguments.length ? (interpolator = _, scale) : interpolator;
};
scale.copy = function() {
return sequential(interpolator).domain([x0, x1]).clamp(clamp);
};
return linearish(scale);
}
var xhtml = "http://www.w3.org/1999/xhtml";
var namespaces = {
svg: "http://www.w3.org/2000/svg",
xhtml: xhtml,
xlink: "http://www.w3.org/1999/xlink",
xml: "http://www.w3.org/XML/1998/namespace",
xmlns: "http://www.w3.org/2000/xmlns/"
};
function namespace(name) {
var prefix = name += "", i = prefix.indexOf(":");
if (i >= 0 && (prefix = name.slice(0, i)) !== "xmlns") name = name.slice(i + 1);
return namespaces.hasOwnProperty(prefix) ? {space: namespaces[prefix], local: name} : name;
}
function creatorInherit(name) {
return function() {
var document = this.ownerDocument,
uri = this.namespaceURI;
return uri === xhtml && document.documentElement.namespaceURI === xhtml
? document.createElement(name)
: document.createElementNS(uri, name);
};
}
function creatorFixed(fullname) {
return function() {
return this.ownerDocument.createElementNS(fullname.space, fullname.local);
};
}
function creator(name) {
var fullname = namespace(name);
return (fullname.local
? creatorFixed
: creatorInherit)(fullname);
}
var nextId = 0;
function local() {
return new Local;
}
function Local() {
this._ = "@" + (++nextId).toString(36);
}
Local.prototype = local.prototype = {
constructor: Local,
get: function(node) {
var id = this._;
while (!(id in node)) if (!(node = node.parentNode)) return;
return node[id];
},
set: function(node, value) {
return node[this._] = value;
},
remove: function(node) {
return this._ in node && delete node[this._];
},
toString: function() {
return this._;
}
};
var matcher = function(selector) {
return function() {
return this.matches(selector);
};
};
if (typeof document !== "undefined") {
var element = document.documentElement;
if (!element.matches) {
var vendorMatches = element.webkitMatchesSelector
|| element.msMatchesSelector
|| element.mozMatchesSelector
|| element.oMatchesSelector;
matcher = function(selector) {
return function() {
return vendorMatches.call(this, selector);
};
};
}
}
var matcher$1 = matcher;
var filterEvents = {};
exports.event = null;
if (typeof document !== "undefined") {
var element$1 = document.documentElement;
if (!("onmouseenter" in element$1)) {
filterEvents = {mouseenter: "mouseover", mouseleave: "mouseout"};
}
}
function filterContextListener(listener, index, group) {
listener = contextListener(listener, index, group);
return function(event) {
var related = event.relatedTarget;
if (!related || (related !== this && !(related.compareDocumentPosition(this) & 8))) {
listener.call(this, event);
}
};
}
function contextListener(listener, index, group) {
return function(event1) {
var event0 = exports.event; // Events can be reentrant (e.g., focus).
exports.event = event1;
try {
listener.call(this, this.__data__, index, group);
} finally {
exports.event = event0;
}
};
}
function parseTypenames$1(typenames) {
return typenames.trim().split(/^|\s+/).map(function(t) {
var name = "", i = t.indexOf(".");
if (i >= 0) name = t.slice(i + 1), t = t.slice(0, i);
return {type: t, name: name};
});
}
function onRemove(typename) {
return function() {
var on = this.__on;
if (!on) return;
for (var j = 0, i = -1, m = on.length, o; j < m; ++j) {
if (o = on[j], (!typename.type || o.type === typename.type) && o.name === typename.name) {
this.removeEventListener(o.type, o.listener, o.capture);
} else {
on[++i] = o;
}
}
if (++i) on.length = i;
else delete this.__on;
};
}
function onAdd(typename, value, capture) {
var wrap = filterEvents.hasOwnProperty(typename.type) ? filterContextListener : contextListener;
return function(d, i, group) {
var on = this.__on, o, listener = wrap(value, i, group);
if (on) for (var j = 0, m = on.length; j < m; ++j) {
if ((o = on[j]).type === typename.type && o.name === typename.name) {
this.removeEventListener(o.type, o.listener, o.capture);
this.addEventListener(o.type, o.listener = listener, o.capture = capture);
o.value = value;
return;
}
}
this.addEventListener(typename.type, listener, capture);
o = {type: typename.type, name: typename.name, value: value, listener: listener, capture: capture};
if (!on) this.__on = [o];
else on.push(o);
};
}
function selection_on(typename, value, capture) {
var typenames = parseTypenames$1(typename + ""), i, n = typenames.length, t;
if (arguments.length < 2) {
var on = this.node().__on;
if (on) for (var j = 0, m = on.length, o; j < m; ++j) {
for (i = 0, o = on[j]; i < n; ++i) {
if ((t = typenames[i]).type === o.type && t.name === o.name) {
return o.value;
}
}
}
return;
}
on = value ? onAdd : onRemove;
if (capture == null) capture = false;
for (i = 0; i < n; ++i) this.each(on(typenames[i], value, capture));
return this;
}
function customEvent(event1, listener, that, args) {
var event0 = exports.event;
event1.sourceEvent = exports.event;
exports.event = event1;
try {
return listener.apply(that, args);
} finally {
exports.event = event0;
}
}
function sourceEvent() {
var current = exports.event, source;
while (source = current.sourceEvent) current = source;
return current;
}
function point$5(node, event) {
var svg = node.ownerSVGElement || node;
if (svg.createSVGPoint) {
var point = svg.createSVGPoint();
point.x = event.clientX, point.y = event.clientY;
point = point.matrixTransform(node.getScreenCTM().inverse());
return [point.x, point.y];
}
var rect = node.getBoundingClientRect();
return [event.clientX - rect.left - node.clientLeft, event.clientY - rect.top - node.clientTop];
}
function mouse(node) {
var event = sourceEvent();
if (event.changedTouches) event = event.changedTouches[0];
return point$5(node, event);
}
function none$2() {}
function selector(selector) {
return selector == null ? none$2 : function() {
return this.querySelector(selector);
};
}
function selection_select(select) {
if (typeof select !== "function") select = selector(select);
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
if ("__data__" in node) subnode.__data__ = node.__data__;
subgroup[i] = subnode;
}
}
}
return new Selection(subgroups, this._parents);
}
function empty() {
return [];
}
function selectorAll(selector) {
return selector == null ? empty : function() {
return this.querySelectorAll(selector);
};
}
function selection_selectAll(select) {
if (typeof select !== "function") select = selectorAll(select);
for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
if (node = group[i]) {
subgroups.push(select.call(node, node.__data__, i, group));
parents.push(node);
}
}
}
return new Selection(subgroups, parents);
}
function selection_filter(match) {
if (typeof match !== "function") match = matcher$1(match);
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
subgroup.push(node);
}
}
}
return new Selection(subgroups, this._parents);
}
function sparse(update) {
return new Array(update.length);
}
function selection_enter() {
return new Selection(this._enter || this._groups.map(sparse), this._parents);
}
function EnterNode(parent, datum) {
this.ownerDocument = parent.ownerDocument;
this.namespaceURI = parent.namespaceURI;
this._next = null;
this._parent = parent;
this.__data__ = datum;
}
EnterNode.prototype = {
constructor: EnterNode,
appendChild: function(child) { return this._parent.insertBefore(child, this._next); },
insertBefore: function(child, next) { return this._parent.insertBefore(child, next); },
querySelector: function(selector) { return this._parent.querySelector(selector); },
querySelectorAll: function(selector) { return this._parent.querySelectorAll(selector); }
};
function constant$4(x) {
return function() {
return x;
};
}
var keyPrefix = "$"; // Protect against keys like “__proto__”.
function bindIndex(parent, group, enter, update, exit, data) {
var i = 0,
node,
groupLength = group.length,
dataLength = data.length;
// Put any non-null nodes that fit into update.
// Put any null nodes into enter.
// Put any remaining data into enter.
for (; i < dataLength; ++i) {
if (node = group[i]) {
node.__data__ = data[i];
update[i] = node;
} else {
enter[i] = new EnterNode(parent, data[i]);
}
}
// Put any non-null nodes that don’t fit into exit.
for (; i < groupLength; ++i) {
if (node = group[i]) {
exit[i] = node;
}
}
}
function bindKey(parent, group, enter, update, exit, data, key) {
var i,
node,
nodeByKeyValue = {},
groupLength = group.length,
dataLength = data.length,
keyValues = new Array(groupLength),
keyValue;
// Compute the key for each node.
// If multiple nodes have the same key, the duplicates are added to exit.
for (i = 0; i < groupLength; ++i) {
if (node = group[i]) {
keyValues[i] = keyValue = keyPrefix + key.call(node, node.__data__, i, group);
if (keyValue in nodeByKeyValue) {
exit[i] = node;
} else {
nodeByKeyValue[keyValue] = node;
}
}
}
// Compute the key for each datum.
// If there a node associated with this key, join and add it to update.
// If there is not (or the key is a duplicate), add it to enter.
for (i = 0; i < dataLength; ++i) {
keyValue = keyPrefix + key.call(parent, data[i], i, data);
if (node = nodeByKeyValue[keyValue]) {
update[i] = node;
node.__data__ = data[i];
nodeByKeyValue[keyValue] = null;
} else {
enter[i] = new EnterNode(parent, data[i]);
}
}
// Add any remaining nodes that were not bound to data to exit.
for (i = 0; i < groupLength; ++i) {
if ((node = group[i]) && (nodeByKeyValue[keyValues[i]] === node)) {
exit[i] = node;
}
}
}
function selection_data(value, key) {
if (!value) {
data = new Array(this.size()), j = -1;
this.each(function(d) { data[++j] = d; });
return data;
}
var bind = key ? bindKey : bindIndex,
parents = this._parents,
groups = this._groups;
if (typeof value !== "function") value = constant$4(value);
for (var m = groups.length, update = new Array(m), enter = new Array(m), exit = new Array(m), j = 0; j < m; ++j) {
var parent = parents[j],
group = groups[j],
groupLength = group.length,
data = value.call(parent, parent && parent.__data__, j, parents),
dataLength = data.length,
enterGroup = enter[j] = new Array(dataLength),
updateGroup = update[j] = new Array(dataLength),
exitGroup = exit[j] = new Array(groupLength);
bind(parent, group, enterGroup, updateGroup, exitGroup, data, key);
// Now connect the enter nodes to their following update node, such that
// appendChild can insert the materialized enter node before this node,
// rather than at the end of the parent node.
for (var i0 = 0, i1 = 0, previous, next; i0 < dataLength; ++i0) {
if (previous = enterGroup[i0]) {
if (i0 >= i1) i1 = i0 + 1;
while (!(next = updateGroup[i1]) && ++i1 < dataLength);
previous._next = next || null;
}
}
}
update = new Selection(update, parents);
update._enter = enter;
update._exit = exit;
return update;
}
function selection_exit() {
return new Selection(this._exit || this._groups.map(sparse), this._parents);
}
function selection_merge(selection) {
for (var groups0 = this._groups, groups1 = selection._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
if (node = group0[i] || group1[i]) {
merge[i] = node;
}
}
}
for (; j < m0; ++j) {
merges[j] = groups0[j];
}
return new Selection(merges, this._parents);
}
function selection_order() {
for (var groups = this._groups, j = -1, m = groups.length; ++j < m;) {
for (var group = groups[j], i = group.length - 1, next = group[i], node; --i >= 0;) {
if (node = group[i]) {
if (next && next !== node.nextSibling) next.parentNode.insertBefore(node, next);
next = node;
}
}
}
return this;
}
function selection_sort(compare) {
if (!compare) compare = ascending$2;
function compareNode(a, b) {
return a && b ? compare(a.__data__, b.__data__) : !a - !b;
}
for (var groups = this._groups, m = groups.length, sortgroups = new Array(m), j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, sortgroup = sortgroups[j] = new Array(n), node, i = 0; i < n; ++i) {
if (node = group[i]) {
sortgroup[i] = node;
}
}
sortgroup.sort(compareNode);
}
return new Selection(sortgroups, this._parents).order();
}
function ascending$2(a, b) {
return a < b ? -1 : a > b ? 1 : a >= b ? 0 : NaN;
}
function selection_call() {
var callback = arguments[0];
arguments[0] = this;
callback.apply(null, arguments);
return this;
}
function selection_nodes() {
var nodes = new Array(this.size()), i = -1;
this.each(function() { nodes[++i] = this; });
return nodes;
}
function selection_node() {
for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
for (var group = groups[j], i = 0, n = group.length; i < n; ++i) {
var node = group[i];
if (node) return node;
}
}
return null;
}
function selection_size() {
var size = 0;
this.each(function() { ++size; });
return size;
}
function selection_empty() {
return !this.node();
}
function selection_each(callback) {
for (var groups = this._groups, j = 0, m = groups.length; j < m; ++j) {
for (var group = groups[j], i = 0, n = group.length, node; i < n; ++i) {
if (node = group[i]) callback.call(node, node.__data__, i, group);
}
}
return this;
}
function attrRemove(name) {
return function() {
this.removeAttribute(name);
};
}
function attrRemoveNS(fullname) {
return function() {
this.removeAttributeNS(fullname.space, fullname.local);
};
}
function attrConstant(name, value) {
return function() {
this.setAttribute(name, value);
};
}
function attrConstantNS(fullname, value) {
return function() {
this.setAttributeNS(fullname.space, fullname.local, value);
};
}
function attrFunction(name, value) {
return function() {
var v = value.apply(this, arguments);
if (v == null) this.removeAttribute(name);
else this.setAttribute(name, v);
};
}
function attrFunctionNS(fullname, value) {
return function() {
var v = value.apply(this, arguments);
if (v == null) this.removeAttributeNS(fullname.space, fullname.local);
else this.setAttributeNS(fullname.space, fullname.local, v);
};
}
function selection_attr(name, value) {
var fullname = namespace(name);
if (arguments.length < 2) {
var node = this.node();
return fullname.local
? node.getAttributeNS(fullname.space, fullname.local)
: node.getAttribute(fullname);
}
return this.each((value == null
? (fullname.local ? attrRemoveNS : attrRemove) : (typeof value === "function"
? (fullname.local ? attrFunctionNS : attrFunction)
: (fullname.local ? attrConstantNS : attrConstant)))(fullname, value));
}
function window$1(node) {
return (node.ownerDocument && node.ownerDocument.defaultView) // node is a Node
|| (node.document && node) // node is a Window
|| node.defaultView; // node is a Document
}
function styleRemove(name) {
return function() {
this.style.removeProperty(name);
};
}
function styleConstant(name, value, priority) {
return function() {
this.style.setProperty(name, value, priority);
};
}
function styleFunction(name, value, priority) {
return function() {
var v = value.apply(this, arguments);
if (v == null) this.style.removeProperty(name);
else this.style.setProperty(name, v, priority);
};
}
function selection_style(name, value, priority) {
var node;
return arguments.length > 1
? this.each((value == null
? styleRemove : typeof value === "function"
? styleFunction
: styleConstant)(name, value, priority == null ? "" : priority))
: window$1(node = this.node())
.getComputedStyle(node, null)
.getPropertyValue(name);
}
function propertyRemove(name) {
return function() {
delete this[name];
};
}
function propertyConstant(name, value) {
return function() {
this[name] = value;
};
}
function propertyFunction(name, value) {
return function() {
var v = value.apply(this, arguments);
if (v == null) delete this[name];
else this[name] = v;
};
}
function selection_property(name, value) {
return arguments.length > 1
? this.each((value == null
? propertyRemove : typeof value === "function"
? propertyFunction
: propertyConstant)(name, value))
: this.node()[name];
}
function classArray(string) {
return string.trim().split(/^|\s+/);
}
function classList(node) {
return node.classList || new ClassList(node);
}
function ClassList(node) {
this._node = node;
this._names = classArray(node.getAttribute("class") || "");
}
ClassList.prototype = {
add: function(name) {
var i = this._names.indexOf(name);
if (i < 0) {
this._names.push(name);
this._node.setAttribute("class", this._names.join(" "));
}
},
remove: function(name) {
var i = this._names.indexOf(name);
if (i >= 0) {
this._names.splice(i, 1);
this._node.setAttribute("class", this._names.join(" "));
}
},
contains: function(name) {
return this._names.indexOf(name) >= 0;
}
};
function classedAdd(node, names) {
var list = classList(node), i = -1, n = names.length;
while (++i < n) list.add(names[i]);
}
function classedRemove(node, names) {
var list = classList(node), i = -1, n = names.length;
while (++i < n) list.remove(names[i]);
}
function classedTrue(names) {
return function() {
classedAdd(this, names);
};
}
function classedFalse(names) {
return function() {
classedRemove(this, names);
};
}
function classedFunction(names, value) {
return function() {
(value.apply(this, arguments) ? classedAdd : classedRemove)(this, names);
};
}
function selection_classed(name, value) {
var names = classArray(name + "");
if (arguments.length < 2) {
var list = classList(this.node()), i = -1, n = names.length;
while (++i < n) if (!list.contains(names[i])) return false;
return true;
}
return this.each((typeof value === "function"
? classedFunction : value
? classedTrue
: classedFalse)(names, value));
}
function textRemove() {
this.textContent = "";
}
function textConstant(value) {
return function() {
this.textContent = value;
};
}
function textFunction(value) {
return function() {
var v = value.apply(this, arguments);
this.textContent = v == null ? "" : v;
};
}
function selection_text(value) {
return arguments.length
? this.each(value == null
? textRemove : (typeof value === "function"
? textFunction
: textConstant)(value))
: this.node().textContent;
}
function htmlRemove() {
this.innerHTML = "";
}
function htmlConstant(value) {
return function() {
this.innerHTML = value;
};
}
function htmlFunction(value) {
return function() {
var v = value.apply(this, arguments);
this.innerHTML = v == null ? "" : v;
};
}
function selection_html(value) {
return arguments.length
? this.each(value == null
? htmlRemove : (typeof value === "function"
? htmlFunction
: htmlConstant)(value))
: this.node().innerHTML;
}
function raise$1() {
if (this.nextSibling) this.parentNode.appendChild(this);
}
function selection_raise() {
return this.each(raise$1);
}
function lower() {
if (this.previousSibling) this.parentNode.insertBefore(this, this.parentNode.firstChild);
}
function selection_lower() {
return this.each(lower);
}
function selection_append(name) {
var create = typeof name === "function" ? name : creator(name);
return this.select(function() {
return this.appendChild(create.apply(this, arguments));
});
}
function constantNull() {
return null;
}
function selection_insert(name, before) {
var create = typeof name === "function" ? name : creator(name),
select = before == null ? constantNull : typeof before === "function" ? before : selector(before);
return this.select(function() {
return this.insertBefore(create.apply(this, arguments), select.apply(this, arguments) || null);
});
}
function remove() {
var parent = this.parentNode;
if (parent) parent.removeChild(this);
}
function selection_remove() {
return this.each(remove);
}
function selection_datum(value) {
return arguments.length
? this.property("__data__", value)
: this.node().__data__;
}
function dispatchEvent(node, type, params) {
var window = window$1(node),
event = window.CustomEvent;
if (event) {
event = new event(type, params);
} else {
event = window.document.createEvent("Event");
if (params) event.initEvent(type, params.bubbles, params.cancelable), event.detail = params.detail;
else event.initEvent(type, false, false);
}
node.dispatchEvent(event);
}
function dispatchConstant(type, params) {
return function() {
return dispatchEvent(this, type, params);
};
}
function dispatchFunction(type, params) {
return function() {
return dispatchEvent(this, type, params.apply(this, arguments));
};
}
function selection_dispatch(type, params) {
return this.each((typeof params === "function"
? dispatchFunction
: dispatchConstant)(type, params));
}
var root = [null];
function Selection(groups, parents) {
this._groups = groups;
this._parents = parents;
}
function selection() {
return new Selection([[document.documentElement]], root);
}
Selection.prototype = selection.prototype = {
constructor: Selection,
select: selection_select,
selectAll: selection_selectAll,
filter: selection_filter,
data: selection_data,
enter: selection_enter,
exit: selection_exit,
merge: selection_merge,
order: selection_order,
sort: selection_sort,
call: selection_call,
nodes: selection_nodes,
node: selection_node,
size: selection_size,
empty: selection_empty,
each: selection_each,
attr: selection_attr,
style: selection_style,
property: selection_property,
classed: selection_classed,
text: selection_text,
html: selection_html,
raise: selection_raise,
lower: selection_lower,
append: selection_append,
insert: selection_insert,
remove: selection_remove,
datum: selection_datum,
on: selection_on,
dispatch: selection_dispatch
};
function select(selector) {
return typeof selector === "string"
? new Selection([[document.querySelector(selector)]], [document.documentElement])
: new Selection([[selector]], root);
}
function selectAll(selector) {
return typeof selector === "string"
? new Selection([document.querySelectorAll(selector)], [document.documentElement])
: new Selection([selector == null ? [] : selector], root);
}
function touch(node, touches, identifier) {
if (arguments.length < 3) identifier = touches, touches = sourceEvent().changedTouches;
for (var i = 0, n = touches ? touches.length : 0, touch; i < n; ++i) {
if ((touch = touches[i]).identifier === identifier) {
return point$5(node, touch);
}
}
return null;
}
function touches(node, touches) {
if (touches == null) touches = sourceEvent().touches;
for (var i = 0, n = touches ? touches.length : 0, points = new Array(n); i < n; ++i) {
points[i] = point$5(node, touches[i]);
}
return points;
}
var emptyOn = dispatch("start", "end", "interrupt");
var emptyTween = [];
var CREATED = 0;
var SCHEDULED = 1;
var STARTING = 2;
var STARTED = 3;
var ENDING = 4;
var ENDED = 5;
function schedule(node, name, id, index, group, timing) {
var schedules = node.__transition;
if (!schedules) node.__transition = {};
else if (id in schedules) return;
create(node, id, {
name: name,
index: index, // For context during callback.
group: group, // For context during callback.
on: emptyOn,
tween: emptyTween,
time: timing.time,
delay: timing.delay,
duration: timing.duration,
ease: timing.ease,
timer: null,
state: CREATED
});
}
function init(node, id) {
var schedule = node.__transition;
if (!schedule || !(schedule = schedule[id]) || schedule.state > CREATED) throw new Error("too late");
return schedule;
}
function set$2(node, id) {
var schedule = node.__transition;
if (!schedule || !(schedule = schedule[id]) || schedule.state > STARTING) throw new Error("too late");
return schedule;
}
function get$1(node, id) {
var schedule = node.__transition;
if (!schedule || !(schedule = schedule[id])) throw new Error("too late");
return schedule;
}
function create(node, id, self) {
var schedules = node.__transition,
tween;
// Initialize the self timer when the transition is created.
// Note the actual delay is not known until the first callback!
schedules[id] = self;
self.timer = timer(schedule, 0, self.time);
// If the delay is greater than this first sleep, sleep some more;
// otherwise, start immediately.
function schedule(elapsed) {
self.state = SCHEDULED;
if (self.delay <= elapsed) start(elapsed - self.delay);
else self.timer.restart(start, self.delay, self.time);
}
function start(elapsed) {
var i, j, n, o;
for (i in schedules) {
o = schedules[i];
if (o.name !== self.name) continue;
// Interrupt the active transition, if any.
// Dispatch the interrupt event.
if (o.state === STARTED) {
o.state = ENDED;
o.timer.stop();
o.on.call("interrupt", node, node.__data__, o.index, o.group);
delete schedules[i];
}
// Cancel any pre-empted transitions. No interrupt event is dispatched
// because the cancelled transitions never started. Note that this also
// removes this transition from the pending list!
else if (+i < id) {
o.state = ENDED;
o.timer.stop();
delete schedules[i];
}
}
// Defer the first tick to end of the current frame; see mbostock/d3#1576.
// Note the transition may be canceled after start and before the first tick!
// Note this must be scheduled before the start event; see d3/d3-transition#16!
// Assuming this is successful, subsequent callbacks go straight to tick.
timeout$1(function() {
if (self.state === STARTED) {
self.timer.restart(tick, self.delay, self.time);
tick(elapsed);
}
});
// Dispatch the start event.
// Note this must be done before the tween are initialized.
self.state = STARTING;
self.on.call("start", node, node.__data__, self.index, self.group);
if (self.state !== STARTING) return; // interrupted
self.state = STARTED;
// Initialize the tween, deleting null tween.
tween = new Array(n = self.tween.length);
for (i = 0, j = -1; i < n; ++i) {
if (o = self.tween[i].value.call(node, node.__data__, self.index, self.group)) {
tween[++j] = o;
}
}
tween.length = j + 1;
}
function tick(elapsed) {
var t = elapsed < self.duration ? self.ease.call(null, elapsed / self.duration) : (self.state = ENDING, 1),
i = -1,
n = tween.length;
while (++i < n) {
tween[i].call(null, t);
}
// Dispatch the end event.
if (self.state === ENDING) {
self.state = ENDED;
self.timer.stop();
self.on.call("end", node, node.__data__, self.index, self.group);
for (i in schedules) if (+i !== id) return void delete schedules[id];
delete node.__transition;
}
}
}
function interrupt(node, name) {
var schedules = node.__transition,
schedule,
active,
empty = true,
i;
if (!schedules) return;
name = name == null ? null : name + "";
for (i in schedules) {
if ((schedule = schedules[i]).name !== name) { empty = false; continue; }
active = schedule.state === STARTED;
schedule.state = ENDED;
schedule.timer.stop();
if (active) schedule.on.call("interrupt", node, node.__data__, schedule.index, schedule.group);
delete schedules[i];
}
if (empty) delete node.__transition;
}
function selection_interrupt(name) {
return this.each(function() {
interrupt(this, name);
});
}
function tweenRemove(id, name) {
var tween0, tween1;
return function() {
var schedule = set$2(this, id),
tween = schedule.tween;
// If this node shared tween with the previous node,
// just assign the updated shared tween and we’re done!
// Otherwise, copy-on-write.
if (tween !== tween0) {
tween1 = tween0 = tween;
for (var i = 0, n = tween1.length; i < n; ++i) {
if (tween1[i].name === name) {
tween1 = tween1.slice();
tween1.splice(i, 1);
break;
}
}
}
schedule.tween = tween1;
};
}
function tweenFunction(id, name, value) {
var tween0, tween1;
if (typeof value !== "function") throw new Error;
return function() {
var schedule = set$2(this, id),
tween = schedule.tween;
// If this node shared tween with the previous node,
// just assign the updated shared tween and we’re done!
// Otherwise, copy-on-write.
if (tween !== tween0) {
tween1 = (tween0 = tween).slice();
for (var t = {name: name, value: value}, i = 0, n = tween1.length; i < n; ++i) {
if (tween1[i].name === name) {
tween1[i] = t;
break;
}
}
if (i === n) tween1.push(t);
}
schedule.tween = tween1;
};
}
function transition_tween(name, value) {
var id = this._id;
name += "";
if (arguments.length < 2) {
var tween = get$1(this.node(), id).tween;
for (var i = 0, n = tween.length, t; i < n; ++i) {
if ((t = tween[i]).name === name) {
return t.value;
}
}
return null;
}
return this.each((value == null ? tweenRemove : tweenFunction)(id, name, value));
}
function tweenValue(transition, name, value) {
var id = transition._id;
transition.each(function() {
var schedule = set$2(this, id);
(schedule.value || (schedule.value = {}))[name] = value.apply(this, arguments);
});
return function(node) {
return get$1(node, id).value[name];
};
}
function interpolate$1(a, b) {
var c;
return (typeof b === "number" ? interpolateNumber
: b instanceof color ? interpolateRgb
: (c = color(b)) ? (b = c, interpolateRgb)
: interpolateString)(a, b);
}
function attrRemove$1(name) {
return function() {
this.removeAttribute(name);
};
}
function attrRemoveNS$1(fullname) {
return function() {
this.removeAttributeNS(fullname.space, fullname.local);
};
}
function attrConstant$1(name, interpolate, value1) {
var value00,
interpolate0;
return function() {
var value0 = this.getAttribute(name);
return value0 === value1 ? null
: value0 === value00 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value1);
};
}
function attrConstantNS$1(fullname, interpolate, value1) {
var value00,
interpolate0;
return function() {
var value0 = this.getAttributeNS(fullname.space, fullname.local);
return value0 === value1 ? null
: value0 === value00 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value1);
};
}
function attrFunction$1(name, interpolate, value) {
var value00,
value10,
interpolate0;
return function() {
var value0, value1 = value(this);
if (value1 == null) return void this.removeAttribute(name);
value0 = this.getAttribute(name);
return value0 === value1 ? null
: value0 === value00 && value1 === value10 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value10 = value1);
};
}
function attrFunctionNS$1(fullname, interpolate, value) {
var value00,
value10,
interpolate0;
return function() {
var value0, value1 = value(this);
if (value1 == null) return void this.removeAttributeNS(fullname.space, fullname.local);
value0 = this.getAttributeNS(fullname.space, fullname.local);
return value0 === value1 ? null
: value0 === value00 && value1 === value10 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value10 = value1);
};
}
function transition_attr(name, value) {
var fullname = namespace(name), i = fullname === "transform" ? interpolateTransform$2 : interpolate$1;
return this.attrTween(name, typeof value === "function"
? (fullname.local ? attrFunctionNS$1 : attrFunction$1)(fullname, i, tweenValue(this, "attr." + name, value))
: value == null ? (fullname.local ? attrRemoveNS$1 : attrRemove$1)(fullname)
: (fullname.local ? attrConstantNS$1 : attrConstant$1)(fullname, i, value));
}
function attrTweenNS(fullname, value) {
function tween() {
var node = this, i = value.apply(node, arguments);
return i && function(t) {
node.setAttributeNS(fullname.space, fullname.local, i(t));
};
}
tween._value = value;
return tween;
}
function attrTween(name, value) {
function tween() {
var node = this, i = value.apply(node, arguments);
return i && function(t) {
node.setAttribute(name, i(t));
};
}
tween._value = value;
return tween;
}
function transition_attrTween(name, value) {
var key = "attr." + name;
if (arguments.length < 2) return (key = this.tween(key)) && key._value;
if (value == null) return this.tween(key, null);
if (typeof value !== "function") throw new Error;
var fullname = namespace(name);
return this.tween(key, (fullname.local ? attrTweenNS : attrTween)(fullname, value));
}
function delayFunction(id, value) {
return function() {
init(this, id).delay = +value.apply(this, arguments);
};
}
function delayConstant(id, value) {
return value = +value, function() {
init(this, id).delay = value;
};
}
function transition_delay(value) {
var id = this._id;
return arguments.length
? this.each((typeof value === "function"
? delayFunction
: delayConstant)(id, value))
: get$1(this.node(), id).delay;
}
function durationFunction(id, value) {
return function() {
set$2(this, id).duration = +value.apply(this, arguments);
};
}
function durationConstant(id, value) {
return value = +value, function() {
set$2(this, id).duration = value;
};
}
function transition_duration(value) {
var id = this._id;
return arguments.length
? this.each((typeof value === "function"
? durationFunction
: durationConstant)(id, value))
: get$1(this.node(), id).duration;
}
function easeConstant(id, value) {
if (typeof value !== "function") throw new Error;
return function() {
set$2(this, id).ease = value;
};
}
function transition_ease(value) {
var id = this._id;
return arguments.length
? this.each(easeConstant(id, value))
: get$1(this.node(), id).ease;
}
function transition_filter(match) {
if (typeof match !== "function") match = matcher$1(match);
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = [], node, i = 0; i < n; ++i) {
if ((node = group[i]) && match.call(node, node.__data__, i, group)) {
subgroup.push(node);
}
}
}
return new Transition(subgroups, this._parents, this._name, this._id);
}
function transition_merge(transition) {
if (transition._id !== this._id) throw new Error;
for (var groups0 = this._groups, groups1 = transition._groups, m0 = groups0.length, m1 = groups1.length, m = Math.min(m0, m1), merges = new Array(m0), j = 0; j < m; ++j) {
for (var group0 = groups0[j], group1 = groups1[j], n = group0.length, merge = merges[j] = new Array(n), node, i = 0; i < n; ++i) {
if (node = group0[i] || group1[i]) {
merge[i] = node;
}
}
}
for (; j < m0; ++j) {
merges[j] = groups0[j];
}
return new Transition(merges, this._parents, this._name, this._id);
}
function start$1(name) {
return (name + "").trim().split(/^|\s+/).every(function(t) {
var i = t.indexOf(".");
if (i >= 0) t = t.slice(0, i);
return !t || t === "start";
});
}
function onFunction(id, name, listener) {
var on0, on1, sit = start$1(name) ? init : set$2;
return function() {
var schedule = sit(this, id),
on = schedule.on;
// If this node shared a dispatch with the previous node,
// just assign the updated shared dispatch and we’re done!
// Otherwise, copy-on-write.
if (on !== on0) (on1 = (on0 = on).copy()).on(name, listener);
schedule.on = on1;
};
}
function transition_on(name, listener) {
var id = this._id;
return arguments.length < 2
? get$1(this.node(), id).on.on(name)
: this.each(onFunction(id, name, listener));
}
function removeFunction(id) {
return function() {
var parent = this.parentNode;
for (var i in this.__transition) if (+i !== id) return;
if (parent) parent.removeChild(this);
};
}
function transition_remove() {
return this.on("end.remove", removeFunction(this._id));
}
function transition_select(select) {
var name = this._name,
id = this._id;
if (typeof select !== "function") select = selector(select);
for (var groups = this._groups, m = groups.length, subgroups = new Array(m), j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, subgroup = subgroups[j] = new Array(n), node, subnode, i = 0; i < n; ++i) {
if ((node = group[i]) && (subnode = select.call(node, node.__data__, i, group))) {
if ("__data__" in node) subnode.__data__ = node.__data__;
subgroup[i] = subnode;
schedule(subgroup[i], name, id, i, subgroup, get$1(node, id));
}
}
}
return new Transition(subgroups, this._parents, name, id);
}
function transition_selectAll(select) {
var name = this._name,
id = this._id;
if (typeof select !== "function") select = selectorAll(select);
for (var groups = this._groups, m = groups.length, subgroups = [], parents = [], j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
if (node = group[i]) {
for (var children = select.call(node, node.__data__, i, group), child, inherit = get$1(node, id), k = 0, l = children.length; k < l; ++k) {
if (child = children[k]) {
schedule(child, name, id, k, children, inherit);
}
}
subgroups.push(children);
parents.push(node);
}
}
}
return new Transition(subgroups, parents, name, id);
}
var Selection$1 = selection.prototype.constructor;
function transition_selection() {
return new Selection$1(this._groups, this._parents);
}
function styleRemove$1(name, interpolate) {
var value00,
value10,
interpolate0;
return function() {
var style = window$1(this).getComputedStyle(this, null),
value0 = style.getPropertyValue(name),
value1 = (this.style.removeProperty(name), style.getPropertyValue(name));
return value0 === value1 ? null
: value0 === value00 && value1 === value10 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value10 = value1);
};
}
function styleRemoveEnd(name) {
return function() {
this.style.removeProperty(name);
};
}
function styleConstant$1(name, interpolate, value1) {
var value00,
interpolate0;
return function() {
var value0 = window$1(this).getComputedStyle(this, null).getPropertyValue(name);
return value0 === value1 ? null
: value0 === value00 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value1);
};
}
function styleFunction$1(name, interpolate, value) {
var value00,
value10,
interpolate0;
return function() {
var style = window$1(this).getComputedStyle(this, null),
value0 = style.getPropertyValue(name),
value1 = value(this);
if (value1 == null) value1 = (this.style.removeProperty(name), style.getPropertyValue(name));
return value0 === value1 ? null
: value0 === value00 && value1 === value10 ? interpolate0
: interpolate0 = interpolate(value00 = value0, value10 = value1);
};
}
function transition_style(name, value, priority) {
var i = (name += "") === "transform" ? interpolateTransform$1 : interpolate$1;
return value == null ? this
.styleTween(name, styleRemove$1(name, i))
.on("end.style." + name, styleRemoveEnd(name))
: this.styleTween(name, typeof value === "function"
? styleFunction$1(name, i, tweenValue(this, "style." + name, value))
: styleConstant$1(name, i, value), priority);
}
function styleTween(name, value, priority) {
function tween() {
var node = this, i = value.apply(node, arguments);
return i && function(t) {
node.style.setProperty(name, i(t), priority);
};
}
tween._value = value;
return tween;
}
function transition_styleTween(name, value, priority) {
var key = "style." + (name += "");
if (arguments.length < 2) return (key = this.tween(key)) && key._value;
if (value == null) return this.tween(key, null);
if (typeof value !== "function") throw new Error;
return this.tween(key, styleTween(name, value, priority == null ? "" : priority));
}
function textConstant$1(value) {
return function() {
this.textContent = value;
};
}
function textFunction$1(value) {
return function() {
var value1 = value(this);
this.textContent = value1 == null ? "" : value1;
};
}
function transition_text(value) {
return this.tween("text", typeof value === "function"
? textFunction$1(tweenValue(this, "text", value))
: textConstant$1(value == null ? "" : value + ""));
}
function transition_transition() {
var name = this._name,
id0 = this._id,
id1 = newId();
for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
if (node = group[i]) {
var inherit = get$1(node, id0);
schedule(node, name, id1, i, group, {
time: inherit.time + inherit.delay + inherit.duration,
delay: 0,
duration: inherit.duration,
ease: inherit.ease
});
}
}
}
return new Transition(groups, this._parents, name, id1);
}
var id = 0;
function Transition(groups, parents, name, id) {
this._groups = groups;
this._parents = parents;
this._name = name;
this._id = id;
}
function transition(name) {
return selection().transition(name);
}
function newId() {
return ++id;
}
var selection_prototype = selection.prototype;
Transition.prototype = transition.prototype = {
constructor: Transition,
select: transition_select,
selectAll: transition_selectAll,
filter: transition_filter,
merge: transition_merge,
selection: transition_selection,
transition: transition_transition,
call: selection_prototype.call,
nodes: selection_prototype.nodes,
node: selection_prototype.node,
size: selection_prototype.size,
empty: selection_prototype.empty,
each: selection_prototype.each,
on: transition_on,
attr: transition_attr,
attrTween: transition_attrTween,
style: transition_style,
styleTween: transition_styleTween,
text: transition_text,
remove: transition_remove,
tween: transition_tween,
delay: transition_delay,
duration: transition_duration,
ease: transition_ease
};
var defaultTiming = {
time: null, // Set on use.
delay: 0,
duration: 250,
ease: easeCubicInOut
};
function inherit(node, id) {
var timing;
while (!(timing = node.__transition) || !(timing = timing[id])) {
if (!(node = node.parentNode)) {
return defaultTiming.time = now(), defaultTiming;
}
}
return timing;
}
function selection_transition(name) {
var id,
timing;
if (name instanceof Transition) {
id = name._id, name = name._name;
} else {
id = newId(), (timing = defaultTiming).time = now(), name = name == null ? null : name + "";
}
for (var groups = this._groups, m = groups.length, j = 0; j < m; ++j) {
for (var group = groups[j], n = group.length, node, i = 0; i < n; ++i) {
if (node = group[i]) {
schedule(node, name, id, i, group, timing || inherit(node, id));
}
}
}
return new Transition(groups, this._parents, name, id);
}
selection.prototype.interrupt = selection_interrupt;
selection.prototype.transition = selection_transition;
var root$1 = [null];
function active(node, name) {
var schedules = node.__transition,
schedule,
i;
if (schedules) {
name = name == null ? null : name + "";
for (i in schedules) {
if ((schedule = schedules[i]).state > SCHEDULED && schedule.name === name) {
return new Transition([[node]], root$1, name, +i);
}
}
}
return null;
}
var slice$4 = Array.prototype.slice;
function identity$5(x) {
return x;
}
var top = 1;
var right = 2;
var bottom = 3;
var left = 4;
var epsilon$2 = 1e-6;
function translateX(scale0, scale1, d) {
var x = scale0(d);
return "translate(" + (isFinite(x) ? x : scale1(d)) + ",0)";
}
function translateY(scale0, scale1, d) {
var y = scale0(d);
return "translate(0," + (isFinite(y) ? y : scale1(d)) + ")";
}
function center(scale) {
var offset = scale.bandwidth() / 2;
if (scale.round()) offset = Math.round(offset);
return function(d) {
return scale(d) + offset;
};
}
function entering() {
return !this.__axis;
}
function axis(orient, scale) {
var tickArguments = [],
tickValues = null,
tickFormat = null,
tickSizeInner = 6,
tickSizeOuter = 6,
tickPadding = 3;
function axis(context) {
var values = tickValues == null ? (scale.ticks ? scale.ticks.apply(scale, tickArguments) : scale.domain()) : tickValues,
format = tickFormat == null ? (scale.tickFormat ? scale.tickFormat.apply(scale, tickArguments) : identity$5) : tickFormat,
spacing = Math.max(tickSizeInner, 0) + tickPadding,
transform = orient === top || orient === bottom ? translateX : translateY,
range = scale.range(),
range0 = range[0] + 0.5,
range1 = range[range.length - 1] + 0.5,
position = (scale.bandwidth ? center : identity$5)(scale.copy()),
selection = context.selection ? context.selection() : context,
path = selection.selectAll(".domain").data([null]),
tick = selection.selectAll(".tick").data(values, scale).order(),
tickExit = tick.exit(),
tickEnter = tick.enter().append("g").attr("class", "tick"),
line = tick.select("line"),
text = tick.select("text"),
k = orient === top || orient === left ? -1 : 1,
x, y = orient === left || orient === right ? (x = "x", "y") : (x = "y", "x");
path = path.merge(path.enter().insert("path", ".tick")
.attr("class", "domain")
.attr("stroke", "#000"));
tick = tick.merge(tickEnter);
line = line.merge(tickEnter.append("line")
.attr("stroke", "#000")
.attr(x + "2", k * tickSizeInner)
.attr(y + "1", 0.5)
.attr(y + "2", 0.5));
text = text.merge(tickEnter.append("text")
.attr("fill", "#000")
.attr(x, k * spacing)
.attr(y, 0.5)
.attr("dy", orient === top ? "0em" : orient === bottom ? "0.71em" : "0.32em"));
if (context !== selection) {
path = path.transition(context);
tick = tick.transition(context);
line = line.transition(context);
text = text.transition(context);
tickExit = tickExit.transition(context)
.attr("opacity", epsilon$2)
.attr("transform", function(d) { return transform(position, this.parentNode.__axis || position, d); });
tickEnter
.attr("opacity", epsilon$2)
.attr("transform", function(d) { return transform(this.parentNode.__axis || position, position, d); });
}
tickExit.remove();
path
.attr("d", orient === left || orient == right
? "M" + k * tickSizeOuter + "," + range0 + "H0.5V" + range1 + "H" + k * tickSizeOuter
: "M" + range0 + "," + k * tickSizeOuter + "V0.5H" + range1 + "V" + k * tickSizeOuter);
tick
.attr("opacity", 1)
.attr("transform", function(d) { return transform(position, position, d); });
line
.attr(x + "2", k * tickSizeInner);
text
.attr(x, k * spacing)
.text(format);
selection.filter(entering)
.attr("fill", "none")
.attr("font-size", 10)
.attr("font-family", "sans-serif")
.attr("text-anchor", orient === right ? "start" : orient === left ? "end" : "middle");
selection
.each(function() { this.__axis = position; });
}
axis.scale = function(_) {
return arguments.length ? (scale = _, axis) : scale;
};
axis.ticks = function() {
return tickArguments = slice$4.call(arguments), axis;
};
axis.tickArguments = function(_) {
return arguments.length ? (tickArguments = _ == null ? [] : slice$4.call(_), axis) : tickArguments.slice();
};
axis.tickValues = function(_) {
return arguments.length ? (tickValues = _ == null ? null : slice$4.call(_), axis) : tickValues && tickValues.slice();
};
axis.tickFormat = function(_) {
return arguments.length ? (tickFormat = _, axis) : tickFormat;
};
axis.tickSize = function(_) {
return arguments.length ? (tickSizeInner = tickSizeOuter = +_, axis) : tickSizeInner;
};
axis.tickSizeInner = function(_) {
return arguments.length ? (tickSizeInner = +_, axis) : tickSizeInner;
};
axis.tickSizeOuter = function(_) {
return arguments.length ? (tickSizeOuter = +_, axis) : tickSizeOuter;
};
axis.tickPadding = function(_) {
return arguments.length ? (tickPadding = +_, axis) : tickPadding;
};
return axis;
}
function axisTop(scale) {
return axis(top, scale);
}
function axisRight(scale) {
return axis(right, scale);
}
function axisBottom(scale) {
return axis(bottom, scale);
}
function axisLeft(scale) {
return axis(left, scale);
}
function defaultSeparation(a, b) {
return a.parent === b.parent ? 1 : 2;
}
function meanX(children) {
return children.reduce(meanXReduce, 0) / children.length;
}
function meanXReduce(x, c) {
return x + c.x;
}
function maxY(children) {
return 1 + children.reduce(maxYReduce, 0);
}
function maxYReduce(y, c) {
return Math.max(y, c.y);
}
function leafLeft(node) {
var children;
while (children = node.children) node = children[0];
return node;
}
function leafRight(node) {
var children;
while (children = node.children) node = children[children.length - 1];
return node;
}
function cluster() {
var separation = defaultSeparation,
dx = 1,
dy = 1,
nodeSize = false;
function cluster(root) {
var previousNode,
x = 0;
// First walk, computing the initial x & y values.
root.eachAfter(function(node) {
var children = node.children;
if (children) {
node.x = meanX(children);
node.y = maxY(children);
} else {
node.x = previousNode ? x += separation(node, previousNode) : 0;
node.y = 0;
previousNode = node;
}
});
var left = leafLeft(root),
right = leafRight(root),
x0 = left.x - separation(left, right) / 2,
x1 = right.x + separation(right, left) / 2;
// Second walk, normalizing x & y to the desired size.
return root.eachAfter(nodeSize ? function(node) {
node.x = (node.x - root.x) * dx;
node.y = (root.y - node.y) * dy;
} : function(node) {
node.x = (node.x - x0) / (x1 - x0) * dx;
node.y = (1 - (root.y ? node.y / root.y : 1)) * dy;
});
}
cluster.separation = function(x) {
return arguments.length ? (separation = x, cluster) : separation;
};
cluster.size = function(x) {
return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? null : [dx, dy]);
};
cluster.nodeSize = function(x) {
return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], cluster) : (nodeSize ? [dx, dy] : null);
};
return cluster;
}
function node_each(callback) {
var node = this, current, next = [node], children, i, n;
do {
current = next.reverse(), next = [];
while (node = current.pop()) {
callback(node), children = node.children;
if (children) for (i = 0, n = children.length; i < n; ++i) {
next.push(children[i]);
}
}
} while (next.length);
return this;
}
function node_eachBefore(callback) {
var node = this, nodes = [node], children, i;
while (node = nodes.pop()) {
callback(node), children = node.children;
if (children) for (i = children.length - 1; i >= 0; --i) {
nodes.push(children[i]);
}
}
return this;
}
function node_eachAfter(callback) {
var node = this, nodes = [node], next = [], children, i, n;
while (node = nodes.pop()) {
next.push(node), children = node.children;
if (children) for (i = 0, n = children.length; i < n; ++i) {
nodes.push(children[i]);
}
}
while (node = next.pop()) {
callback(node);
}
return this;
}
function node_sum(value) {
return this.eachAfter(function(node) {
var sum = +value(node.data) || 0,
children = node.children,
i = children && children.length;
while (--i >= 0) sum += children[i].value;
node.value = sum;
});
}
function node_sort(compare) {
return this.eachBefore(function(node) {
if (node.children) {
node.children.sort(compare);
}
});
}
function node_path(end) {
var start = this,
ancestor = leastCommonAncestor(start, end),
nodes = [start];
while (start !== ancestor) {
start = start.parent;
nodes.push(start);
}
var k = nodes.length;
while (end !== ancestor) {
nodes.splice(k, 0, end);
end = end.parent;
}
return nodes;
}
function leastCommonAncestor(a, b) {
if (a === b) return a;
var aNodes = a.ancestors(),
bNodes = b.ancestors(),
c = null;
a = aNodes.pop();
b = bNodes.pop();
while (a === b) {
c = a;
a = aNodes.pop();
b = bNodes.pop();
}
return c;
}
function node_ancestors() {
var node = this, nodes = [node];
while (node = node.parent) {
nodes.push(node);
}
return nodes;
}
function node_descendants() {
var nodes = [];
this.each(function(node) {
nodes.push(node);
});
return nodes;
}
function node_leaves() {
var leaves = [];
this.eachBefore(function(node) {
if (!node.children) {
leaves.push(node);
}
});
return leaves;
}
function node_links() {
var root = this, links = [];
root.each(function(node) {
if (node !== root) { // Don’t include the root’s parent, if any.
links.push({source: node.parent, target: node});
}
});
return links;
}
function hierarchy(data, children) {
var root = new Node(data),
valued = +data.value && (root.value = data.value),
node,
nodes = [root],
child,
childs,
i,
n;
if (children == null) children = defaultChildren;
while (node = nodes.pop()) {
if (valued) node.value = +node.data.value;
if ((childs = children(node.data)) && (n = childs.length)) {
node.children = new Array(n);
for (i = n - 1; i >= 0; --i) {
nodes.push(child = node.children[i] = new Node(childs[i]));
child.parent = node;
child.depth = node.depth + 1;
}
}
}
return root.eachBefore(computeHeight);
}
function node_copy() {
return hierarchy(this).eachBefore(copyData);
}
function defaultChildren(d) {
return d.children;
}
function copyData(node) {
node.data = node.data.data;
}
function computeHeight(node) {
var height = 0;
do node.height = height;
while ((node = node.parent) && (node.height < ++height));
}
function Node(data) {
this.data = data;
this.depth =
this.height = 0;
this.parent = null;
}
Node.prototype = hierarchy.prototype = {
constructor: Node,
each: node_each,
eachAfter: node_eachAfter,
eachBefore: node_eachBefore,
sum: node_sum,
sort: node_sort,
path: node_path,
ancestors: node_ancestors,
descendants: node_descendants,
leaves: node_leaves,
links: node_links,
copy: node_copy
};
function Node$2(value) {
this._ = value;
this.next = null;
}
function shuffle$1(array) {
var i,
n = (array = array.slice()).length,
head = null,
node = head;
while (n) {
var next = new Node$2(array[n - 1]);
if (node) node = node.next = next;
else node = head = next;
array[i] = array[--n];
}
return {
head: head,
tail: node
};
}
function enclose(circles) {
return encloseN(shuffle$1(circles), []);
}
function encloses(a, b) {
var dx = b.x - a.x,
dy = b.y - a.y,
dr = a.r - b.r;
return dr * dr + 1e-6 > dx * dx + dy * dy;
}
// Returns the smallest circle that contains circles L and intersects circles B.
function encloseN(L, B) {
var circle,
l0 = null,
l1 = L.head,
l2,
p1;
switch (B.length) {
case 1: circle = enclose1(B[0]); break;
case 2: circle = enclose2(B[0], B[1]); break;
case 3: circle = enclose3(B[0], B[1], B[2]); break;
}
while (l1) {
p1 = l1._, l2 = l1.next;
if (!circle || !encloses(circle, p1)) {
// Temporarily truncate L before l1.
if (l0) L.tail = l0, l0.next = null;
else L.head = L.tail = null;
B.push(p1);
circle = encloseN(L, B); // Note: reorders L!
B.pop();
// Move l1 to the front of L and reconnect the truncated list L.
if (L.head) l1.next = L.head, L.head = l1;
else l1.next = null, L.head = L.tail = l1;
l0 = L.tail, l0.next = l2;
} else {
l0 = l1;
}
l1 = l2;
}
L.tail = l0;
return circle;
}
function enclose1(a) {
return {
x: a.x,
y: a.y,
r: a.r
};
}
function enclose2(a, b) {
var x1 = a.x, y1 = a.y, r1 = a.r,
x2 = b.x, y2 = b.y, r2 = b.r,
x21 = x2 - x1, y21 = y2 - y1, r21 = r2 - r1,
l = Math.sqrt(x21 * x21 + y21 * y21);
return {
x: (x1 + x2 + x21 / l * r21) / 2,
y: (y1 + y2 + y21 / l * r21) / 2,
r: (l + r1 + r2) / 2
};
}
function enclose3(a, b, c) {
var x1 = a.x, y1 = a.y, r1 = a.r,
x2 = b.x, y2 = b.y, r2 = b.r,
x3 = c.x, y3 = c.y, r3 = c.r,
a2 = 2 * (x1 - x2),
b2 = 2 * (y1 - y2),
c2 = 2 * (r2 - r1),
d2 = x1 * x1 + y1 * y1 - r1 * r1 - x2 * x2 - y2 * y2 + r2 * r2,
a3 = 2 * (x1 - x3),
b3 = 2 * (y1 - y3),
c3 = 2 * (r3 - r1),
d3 = x1 * x1 + y1 * y1 - r1 * r1 - x3 * x3 - y3 * y3 + r3 * r3,
ab = a3 * b2 - a2 * b3,
xa = (b2 * d3 - b3 * d2) / ab - x1,
xb = (b3 * c2 - b2 * c3) / ab,
ya = (a3 * d2 - a2 * d3) / ab - y1,
yb = (a2 * c3 - a3 * c2) / ab,
A = xb * xb + yb * yb - 1,
B = 2 * (xa * xb + ya * yb + r1),
C = xa * xa + ya * ya - r1 * r1,
r = (-B - Math.sqrt(B * B - 4 * A * C)) / (2 * A);
return {
x: xa + xb * r + x1,
y: ya + yb * r + y1,
r: r
};
}
function place(a, b, c) {
var ax = a.x,
ay = a.y,
da = b.r + c.r,
db = a.r + c.r,
dx = b.x - ax,
dy = b.y - ay,
dc = dx * dx + dy * dy;
if (dc) {
var x = 0.5 + ((db *= db) - (da *= da)) / (2 * dc),
y = Math.sqrt(Math.max(0, 2 * da * (db + dc) - (db -= dc) * db - da * da)) / (2 * dc);
c.x = ax + x * dx + y * dy;
c.y = ay + x * dy - y * dx;
} else {
c.x = ax + db;
c.y = ay;
}
}
function intersects(a, b) {
var dx = b.x - a.x,
dy = b.y - a.y,
dr = a.r + b.r;
return dr * dr > dx * dx + dy * dy;
}
function distance2(circle, x, y) {
var dx = circle.x - x,
dy = circle.y - y;
return dx * dx + dy * dy;
}
function Node$1(circle) {
this._ = circle;
this.next = null;
this.previous = null;
}
function packEnclose(circles) {
if (!(n = circles.length)) return 0;
var a, b, c, n;
// Place the first circle.
a = circles[0], a.x = 0, a.y = 0;
if (!(n > 1)) return a.r;
// Place the second circle.
b = circles[1], a.x = -b.r, b.x = a.r, b.y = 0;
if (!(n > 2)) return a.r + b.r;
// Place the third circle.
place(b, a, c = circles[2]);
// Initialize the weighted centroid.
var aa = a.r * a.r,
ba = b.r * b.r,
ca = c.r * c.r,
oa = aa + ba + ca,
ox = aa * a.x + ba * b.x + ca * c.x,
oy = aa * a.y + ba * b.y + ca * c.y,
cx, cy, i, j, k, sj, sk;
// Initialize the front-chain using the first three circles a, b and c.
a = new Node$1(a), b = new Node$1(b), c = new Node$1(c);
a.next = c.previous = b;
b.next = a.previous = c;
c.next = b.previous = a;
// Attempt to place each remaining circle…
pack: for (i = 3; i < n; ++i) {
place(a._, b._, c = circles[i]), c = new Node$1(c);
// If there are only three elements in the front-chain…
if ((k = a.previous) === (j = b.next)) {
// If the new circle intersects the third circle,
// rotate the front chain to try the next position.
if (intersects(j._, c._)) {
a = b, b = j, --i;
continue pack;
}
}
// Find the closest intersecting circle on the front-chain, if any.
else {
sj = j._.r, sk = k._.r;
do {
if (sj <= sk) {
if (intersects(j._, c._)) {
b = j, a.next = b, b.previous = a, --i;
continue pack;
}
j = j.next, sj += j._.r;
} else {
if (intersects(k._, c._)) {
a = k, a.next = b, b.previous = a, --i;
continue pack;
}
k = k.previous, sk += k._.r;
}
} while (j !== k.next);
}
// Success! Insert the new circle c between a and b.
c.previous = a, c.next = b, a.next = b.previous = b = c;
// Update the weighted centroid.
oa += ca = c._.r * c._.r;
ox += ca * c._.x;
oy += ca * c._.y;
// Compute the new closest circle a to centroid.
aa = distance2(a._, cx = ox / oa, cy = oy / oa);
while ((c = c.next) !== b) {
if ((ca = distance2(c._, cx, cy)) < aa) {
a = c, aa = ca;
}
}
b = a.next;
}
// Compute the enclosing circle of the front chain.
a = [b._], c = b; while ((c = c.next) !== b) a.push(c._); c = enclose(a);
// Translate the circles to put the enclosing circle around the origin.
for (i = 0; i < n; ++i) a = circles[i], a.x -= c.x, a.y -= c.y;
return c.r;
}
function siblings(circles) {
packEnclose(circles);
return circles;
}
function optional(f) {
return f == null ? null : required(f);
}
function required(f) {
if (typeof f !== "function") throw new Error;
return f;
}
function constantZero() {
return 0;
}
function constant$5(x) {
return function() {
return x;
};
}
function defaultRadius(d) {
return Math.sqrt(d.value);
}
function index() {
var radius = null,
dx = 1,
dy = 1,
padding = constantZero;
function pack(root) {
root.x = dx / 2, root.y = dy / 2;
if (radius) {
root.eachBefore(radiusLeaf(radius))
.eachAfter(packChildren(padding, 0.5))
.eachBefore(translateChild(1));
} else {
root.eachBefore(radiusLeaf(defaultRadius))
.eachAfter(packChildren(constantZero, 1))
.eachAfter(packChildren(padding, root.r / Math.min(dx, dy)))
.eachBefore(translateChild(Math.min(dx, dy) / (2 * root.r)));
}
return root;
}
pack.radius = function(x) {
return arguments.length ? (radius = optional(x), pack) : radius;
};
pack.size = function(x) {
return arguments.length ? (dx = +x[0], dy = +x[1], pack) : [dx, dy];
};
pack.padding = function(x) {
return arguments.length ? (padding = typeof x === "function" ? x : constant$5(+x), pack) : padding;
};
return pack;
}
function radiusLeaf(radius) {
return function(node) {
if (!node.children) {
node.r = Math.max(0, +radius(node) || 0);
}
};
}
function packChildren(padding, k) {
return function(node) {
if (children = node.children) {
var children,
i,
n = children.length,
r = padding(node) * k || 0,
e;
if (r) for (i = 0; i < n; ++i) children[i].r += r;
e = packEnclose(children);
if (r) for (i = 0; i < n; ++i) children[i].r -= r;
node.r = e + r;
}
};
}
function translateChild(k) {
return function(node) {
var parent = node.parent;
node.r *= k;
if (parent) {
node.x = parent.x + k * node.x;
node.y = parent.y + k * node.y;
}
};
}
function roundNode(node) {
node.x0 = Math.round(node.x0);
node.y0 = Math.round(node.y0);
node.x1 = Math.round(node.x1);
node.y1 = Math.round(node.y1);
}
function treemapDice(parent, x0, y0, x1, y1) {
var nodes = parent.children,
node,
i = -1,
n = nodes.length,
k = parent.value && (x1 - x0) / parent.value;
while (++i < n) {
node = nodes[i], node.y0 = y0, node.y1 = y1;
node.x0 = x0, node.x1 = x0 += node.value * k;
}
}
function partition() {
var dx = 1,
dy = 1,
padding = 0,
round = false;
function partition(root) {
var n = root.height + 1;
root.x0 =
root.y0 = padding;
root.x1 = dx;
root.y1 = dy / n;
root.eachBefore(positionNode(dy, n));
if (round) root.eachBefore(roundNode);
return root;
}
function positionNode(dy, n) {
return function(node) {
if (node.children) {
treemapDice(node, node.x0, dy * (node.depth + 1) / n, node.x1, dy * (node.depth + 2) / n);
}
var x0 = node.x0,
y0 = node.y0,
x1 = node.x1 - padding,
y1 = node.y1 - padding;
if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
node.x0 = x0;
node.y0 = y0;
node.x1 = x1;
node.y1 = y1;
};
}
partition.round = function(x) {
return arguments.length ? (round = !!x, partition) : round;
};
partition.size = function(x) {
return arguments.length ? (dx = +x[0], dy = +x[1], partition) : [dx, dy];
};
partition.padding = function(x) {
return arguments.length ? (padding = +x, partition) : padding;
};
return partition;
}
var keyPrefix$1 = "$";
var preroot = {depth: -1};
var ambiguous = {};
function defaultId(d) {
return d.id;
}
function defaultParentId(d) {
return d.parentId;
}
function stratify() {
var id = defaultId,
parentId = defaultParentId;
function stratify(data) {
var d,
i,
n = data.length,
root,
parent,
node,
nodes = new Array(n),
nodeId,
nodeKey,
nodeByKey = {};
for (i = 0; i < n; ++i) {
d = data[i], node = nodes[i] = new Node(d);
if ((nodeId = id(d, i, data)) != null && (nodeId += "")) {
nodeKey = keyPrefix$1 + (node.id = nodeId);
nodeByKey[nodeKey] = nodeKey in nodeByKey ? ambiguous : node;
}
}
for (i = 0; i < n; ++i) {
node = nodes[i], nodeId = parentId(data[i], i, data);
if (nodeId == null || !(nodeId += "")) {
if (root) throw new Error("multiple roots");
root = node;
} else {
parent = nodeByKey[keyPrefix$1 + nodeId];
if (!parent) throw new Error("missing: " + nodeId);
if (parent === ambiguous) throw new Error("ambiguous: " + nodeId);
if (parent.children) parent.children.push(node);
else parent.children = [node];
node.parent = parent;
}
}
if (!root) throw new Error("no root");
root.parent = preroot;
root.eachBefore(function(node) { node.depth = node.parent.depth + 1; --n; }).eachBefore(computeHeight);
root.parent = null;
if (n > 0) throw new Error("cycle");
return root;
}
stratify.id = function(x) {
return arguments.length ? (id = required(x), stratify) : id;
};
stratify.parentId = function(x) {
return arguments.length ? (parentId = required(x), stratify) : parentId;
};
return stratify;
}
function defaultSeparation$1(a, b) {
return a.parent === b.parent ? 1 : 2;
}
// function radialSeparation(a, b) {
// return (a.parent === b.parent ? 1 : 2) / a.depth;
// }
// This function is used to traverse the left contour of a subtree (or
// subforest). It returns the successor of v on this contour. This successor is
// either given by the leftmost child of v or by the thread of v. The function
// returns null if and only if v is on the highest level of its subtree.
function nextLeft(v) {
var children = v.children;
return children ? children[0] : v.t;
}
// This function works analogously to nextLeft.
function nextRight(v) {
var children = v.children;
return children ? children[children.length - 1] : v.t;
}
// Shifts the current subtree rooted at w+. This is done by increasing
// prelim(w+) and mod(w+) by shift.
function moveSubtree(wm, wp, shift) {
var change = shift / (wp.i - wm.i);
wp.c -= change;
wp.s += shift;
wm.c += change;
wp.z += shift;
wp.m += shift;
}
// All other shifts, applied to the smaller subtrees between w- and w+, are
// performed by this function. To prepare the shifts, we have to adjust
// change(w+), shift(w+), and change(w-).
function executeShifts(v) {
var shift = 0,
change = 0,
children = v.children,
i = children.length,
w;
while (--i >= 0) {
w = children[i];
w.z += shift;
w.m += shift;
shift += w.s + (change += w.c);
}
}
// If vi-’s ancestor is a sibling of v, returns vi-’s ancestor. Otherwise,
// returns the specified (default) ancestor.
function nextAncestor(vim, v, ancestor) {
return vim.a.parent === v.parent ? vim.a : ancestor;
}
function TreeNode(node, i) {
this._ = node;
this.parent = null;
this.children = null;
this.A = null; // default ancestor
this.a = this; // ancestor
this.z = 0; // prelim
this.m = 0; // mod
this.c = 0; // change
this.s = 0; // shift
this.t = null; // thread
this.i = i; // number
}
TreeNode.prototype = Object.create(Node.prototype);
function treeRoot(root) {
var tree = new TreeNode(root, 0),
node,
nodes = [tree],
child,
children,
i,
n;
while (node = nodes.pop()) {
if (children = node._.children) {
node.children = new Array(n = children.length);
for (i = n - 1; i >= 0; --i) {
nodes.push(child = node.children[i] = new TreeNode(children[i], i));
child.parent = node;
}
}
}
(tree.parent = new TreeNode(null, 0)).children = [tree];
return tree;
}
// Node-link tree diagram using the Reingold-Tilford "tidy" algorithm
function tree() {
var separation = defaultSeparation$1,
dx = 1,
dy = 1,
nodeSize = null;
function tree(root) {
var t = treeRoot(root);
// Compute the layout using Buchheim et al.’s algorithm.
t.eachAfter(firstWalk), t.parent.m = -t.z;
t.eachBefore(secondWalk);
// If a fixed node size is specified, scale x and y.
if (nodeSize) root.eachBefore(sizeNode);
// If a fixed tree size is specified, scale x and y based on the extent.
// Compute the left-most, right-most, and depth-most nodes for extents.
else {
var left = root,
right = root,
bottom = root;
root.eachBefore(function(node) {
if (node.x < left.x) left = node;
if (node.x > right.x) right = node;
if (node.depth > bottom.depth) bottom = node;
});
var s = left === right ? 1 : separation(left, right) / 2,
tx = s - left.x,
kx = dx / (right.x + s + tx),
ky = dy / (bottom.depth || 1);
root.eachBefore(function(node) {
node.x = (node.x + tx) * kx;
node.y = node.depth * ky;
});
}
return root;
}
// Computes a preliminary x-coordinate for v. Before that, FIRST WALK is
// applied recursively to the children of v, as well as the function
// APPORTION. After spacing out the children by calling EXECUTE SHIFTS, the
// node v is placed to the midpoint of its outermost children.
function firstWalk(v) {
var children = v.children,
siblings = v.parent.children,
w = v.i ? siblings[v.i - 1] : null;
if (children) {
executeShifts(v);
var midpoint = (children[0].z + children[children.length - 1].z) / 2;
if (w) {
v.z = w.z + separation(v._, w._);
v.m = v.z - midpoint;
} else {
v.z = midpoint;
}
} else if (w) {
v.z = w.z + separation(v._, w._);
}
v.parent.A = apportion(v, w, v.parent.A || siblings[0]);
}
// Computes all real x-coordinates by summing up the modifiers recursively.
function secondWalk(v) {
v._.x = v.z + v.parent.m;
v.m += v.parent.m;
}
// The core of the algorithm. Here, a new subtree is combined with the
// previous subtrees. Threads are used to traverse the inside and outside
// contours of the left and right subtree up to the highest common level. The
// vertices used for the traversals are vi+, vi-, vo-, and vo+, where the
// superscript o means outside and i means inside, the subscript - means left
// subtree and + means right subtree. For summing up the modifiers along the
// contour, we use respective variables si+, si-, so-, and so+. Whenever two
// nodes of the inside contours conflict, we compute the left one of the
// greatest uncommon ancestors using the function ANCESTOR and call MOVE
// SUBTREE to shift the subtree and prepare the shifts of smaller subtrees.
// Finally, we add a new thread (if necessary).
function apportion(v, w, ancestor) {
if (w) {
var vip = v,
vop = v,
vim = w,
vom = vip.parent.children[0],
sip = vip.m,
sop = vop.m,
sim = vim.m,
som = vom.m,
shift;
while (vim = nextRight(vim), vip = nextLeft(vip), vim && vip) {
vom = nextLeft(vom);
vop = nextRight(vop);
vop.a = v;
shift = vim.z + sim - vip.z - sip + separation(vim._, vip._);
if (shift > 0) {
moveSubtree(nextAncestor(vim, v, ancestor), v, shift);
sip += shift;
sop += shift;
}
sim += vim.m;
sip += vip.m;
som += vom.m;
sop += vop.m;
}
if (vim && !nextRight(vop)) {
vop.t = vim;
vop.m += sim - sop;
}
if (vip && !nextLeft(vom)) {
vom.t = vip;
vom.m += sip - som;
ancestor = v;
}
}
return ancestor;
}
function sizeNode(node) {
node.x *= dx;
node.y = node.depth * dy;
}
tree.separation = function(x) {
return arguments.length ? (separation = x, tree) : separation;
};
tree.size = function(x) {
return arguments.length ? (nodeSize = false, dx = +x[0], dy = +x[1], tree) : (nodeSize ? null : [dx, dy]);
};
tree.nodeSize = function(x) {
return arguments.length ? (nodeSize = true, dx = +x[0], dy = +x[1], tree) : (nodeSize ? [dx, dy] : null);
};
return tree;
}
function treemapSlice(parent, x0, y0, x1, y1) {
var nodes = parent.children,
node,
i = -1,
n = nodes.length,
k = parent.value && (y1 - y0) / parent.value;
while (++i < n) {
node = nodes[i], node.x0 = x0, node.x1 = x1;
node.y0 = y0, node.y1 = y0 += node.value * k;
}
}
var phi = (1 + Math.sqrt(5)) / 2;
function squarifyRatio(ratio, parent, x0, y0, x1, y1) {
var rows = [],
nodes = parent.children,
row,
nodeValue,
i0 = 0,
i1,
n = nodes.length,
dx, dy,
value = parent.value,
sumValue,
minValue,
maxValue,
newRatio,
minRatio,
alpha,
beta;
while (i0 < n) {
dx = x1 - x0, dy = y1 - y0;
minValue = maxValue = sumValue = nodes[i0].value;
alpha = Math.max(dy / dx, dx / dy) / (value * ratio);
beta = sumValue * sumValue * alpha;
minRatio = Math.max(maxValue / beta, beta / minValue);
// Keep adding nodes while the aspect ratio maintains or improves.
for (i1 = i0 + 1; i1 < n; ++i1) {
sumValue += nodeValue = nodes[i1].value;
if (nodeValue < minValue) minValue = nodeValue;
if (nodeValue > maxValue) maxValue = nodeValue;
beta = sumValue * sumValue * alpha;
newRatio = Math.max(maxValue / beta, beta / minValue);
if (newRatio > minRatio) { sumValue -= nodeValue; break; }
minRatio = newRatio;
}
// Position and record the row orientation.
rows.push(row = {value: sumValue, dice: dx < dy, children: nodes.slice(i0, i1)});
if (row.dice) treemapDice(row, x0, y0, x1, value ? y0 += dy * sumValue / value : y1);
else treemapSlice(row, x0, y0, value ? x0 += dx * sumValue / value : x1, y1);
value -= sumValue, i0 = i1;
}
return rows;
}
var squarify = (function custom(ratio) {
function squarify(parent, x0, y0, x1, y1) {
squarifyRatio(ratio, parent, x0, y0, x1, y1);
}
squarify.ratio = function(x) {
return custom((x = +x) > 1 ? x : 1);
};
return squarify;
})(phi);
function index$1() {
var tile = squarify,
round = false,
dx = 1,
dy = 1,
paddingStack = [0],
paddingInner = constantZero,
paddingTop = constantZero,
paddingRight = constantZero,
paddingBottom = constantZero,
paddingLeft = constantZero;
function treemap(root) {
root.x0 =
root.y0 = 0;
root.x1 = dx;
root.y1 = dy;
root.eachBefore(positionNode);
paddingStack = [0];
if (round) root.eachBefore(roundNode);
return root;
}
function positionNode(node) {
var p = paddingStack[node.depth],
x0 = node.x0 + p,
y0 = node.y0 + p,
x1 = node.x1 - p,
y1 = node.y1 - p;
if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
node.x0 = x0;
node.y0 = y0;
node.x1 = x1;
node.y1 = y1;
if (node.children) {
p = paddingStack[node.depth + 1] = paddingInner(node) / 2;
x0 += paddingLeft(node) - p;
y0 += paddingTop(node) - p;
x1 -= paddingRight(node) - p;
y1 -= paddingBottom(node) - p;
if (x1 < x0) x0 = x1 = (x0 + x1) / 2;
if (y1 < y0) y0 = y1 = (y0 + y1) / 2;
tile(node, x0, y0, x1, y1);
}
}
treemap.round = function(x) {
return arguments.length ? (round = !!x, treemap) : round;
};
treemap.size = function(x) {
return arguments.length ? (dx = +x[0], dy = +x[1], treemap) : [dx, dy];
};
treemap.tile = function(x) {
return arguments.length ? (tile = required(x), treemap) : tile;
};
treemap.padding = function(x) {
return arguments.length ? treemap.paddingInner(x).paddingOuter(x) : treemap.paddingInner();
};
treemap.paddingInner = function(x) {
return arguments.length ? (paddingInner = typeof x === "function" ? x : constant$5(+x), treemap) : paddingInner;
};
treemap.paddingOuter = function(x) {
return arguments.length ? treemap.paddingTop(x).paddingRight(x).paddingBottom(x).paddingLeft(x) : treemap.paddingTop();
};
treemap.paddingTop = function(x) {
return arguments.length ? (paddingTop = typeof x === "function" ? x : constant$5(+x), treemap) : paddingTop;
};
treemap.paddingRight = function(x) {
return arguments.length ? (paddingRight = typeof x === "function" ? x : constant$5(+x), treemap) : paddingRight;
};
treemap.paddingBottom = function(x) {
return arguments.length ? (paddingBottom = typeof x === "function" ? x : constant$5(+x), treemap) : paddingBottom;
};
treemap.paddingLeft = function(x) {
return arguments.length ? (paddingLeft = typeof x === "function" ? x : constant$5(+x), treemap) : paddingLeft;
};
return treemap;
}
function binary(parent, x0, y0, x1, y1) {
var nodes = parent.children,
i, n = nodes.length,
sum, sums = new Array(n + 1);
for (sums[0] = sum = i = 0; i < n; ++i) {
sums[i + 1] = sum += nodes[i].value;
}
partition(0, n, parent.value, x0, y0, x1, y1);
function partition(i, j, value, x0, y0, x1, y1) {
if (i >= j - 1) {
var node = nodes[i];
node.x0 = x0, node.y0 = y0;
node.x1 = x1, node.y1 = y1;
return;
}
var valueOffset = sums[i],
valueTarget = (value / 2) + valueOffset,
k = i + 1,
hi = j - 1;
while (k < hi) {
var mid = k + hi >>> 1;
if (sums[mid] < valueTarget) k = mid + 1;
else hi = mid;
}
var valueLeft = sums[k] - valueOffset,
valueRight = value - valueLeft;
if ((y1 - y0) > (x1 - x0)) {
var yk = (y0 * valueRight + y1 * valueLeft) / value;
partition(i, k, valueLeft, x0, y0, x1, yk);
partition(k, j, valueRight, x0, yk, x1, y1);
} else {
var xk = (x0 * valueRight + x1 * valueLeft) / value;
partition(i, k, valueLeft, x0, y0, xk, y1);
partition(k, j, valueRight, xk, y0, x1, y1);
}
}
}
function sliceDice(parent, x0, y0, x1, y1) {
(parent.depth & 1 ? treemapSlice : treemapDice)(parent, x0, y0, x1, y1);
}
var resquarify = (function custom(ratio) {
function resquarify(parent, x0, y0, x1, y1) {
if ((rows = parent._squarify) && (rows.ratio === ratio)) {
var rows,
row,
nodes,
i,
j = -1,
n,
m = rows.length,
value = parent.value;
while (++j < m) {
row = rows[j], nodes = row.children;
for (i = row.value = 0, n = nodes.length; i < n; ++i) row.value += nodes[i].value;
if (row.dice) treemapDice(row, x0, y0, x1, y0 += (y1 - y0) * row.value / value);
else treemapSlice(row, x0, y0, x0 += (x1 - x0) * row.value / value, y1);
value -= row.value;
}
} else {
parent._squarify = rows = squarifyRatio(ratio, parent, x0, y0, x1, y1);
rows.ratio = ratio;
}
}
resquarify.ratio = function(x) {
return custom((x = +x) > 1 ? x : 1);
};
return resquarify;
})(phi);
function center$1(x, y) {
var nodes;
if (x == null) x = 0;
if (y == null) y = 0;
function force() {
var i,
n = nodes.length,
node,
sx = 0,
sy = 0;
for (i = 0; i < n; ++i) {
node = nodes[i], sx += node.x, sy += node.y;
}
for (sx = sx / n - x, sy = sy / n - y, i = 0; i < n; ++i) {
node = nodes[i], node.x -= sx, node.y -= sy;
}
}
force.initialize = function(_) {
nodes = _;
};
force.x = function(_) {
return arguments.length ? (x = +_, force) : x;
};
force.y = function(_) {
return arguments.length ? (y = +_, force) : y;
};
return force;
}
function constant$6(x) {
return function() {
return x;
};
}
function jiggle() {
return (Math.random() - 0.5) * 1e-6;
}
function x$1(d) {
return d.x + d.vx;
}
function y$1(d) {
return d.y + d.vy;
}
function collide(radius) {
var nodes,
radii,
strength = 1,
iterations = 1;
if (typeof radius !== "function") radius = constant$6(radius == null ? 1 : +radius);
function force() {
var i, n = nodes.length,
tree,
node,
xi,
yi,
ri,
ri2;
for (var k = 0; k < iterations; ++k) {
tree = quadtree(nodes, x$1, y$1).visitAfter(prepare);
for (i = 0; i < n; ++i) {
node = nodes[i];
ri = radii[i], ri2 = ri * ri;
xi = node.x + node.vx;
yi = node.y + node.vy;
tree.visit(apply);
}
}
function apply(quad, x0, y0, x1, y1) {
var data = quad.data, rj = quad.r, r = ri + rj;
if (data) {
if (data.index > i) {
var x = xi - data.x - data.vx,
y = yi - data.y - data.vy,
l = x * x + y * y;
if (l < r * r) {
if (x === 0) x = jiggle(), l += x * x;
if (y === 0) y = jiggle(), l += y * y;
l = (r - (l = Math.sqrt(l))) / l * strength;
node.vx += (x *= l) * (r = (rj *= rj) / (ri2 + rj));
node.vy += (y *= l) * r;
data.vx -= x * (r = 1 - r);
data.vy -= y * r;
}
}
return;
}
return x0 > xi + r || x1 < xi - r || y0 > yi + r || y1 < yi - r;
}
}
function prepare(quad) {
if (quad.data) return quad.r = radii[quad.data.index];
for (var i = quad.r = 0; i < 4; ++i) {
if (quad[i] && quad[i].r > quad.r) {
quad.r = quad[i].r;
}
}
}
force.initialize = function(_) {
var i, n = (nodes = _).length; radii = new Array(n);
for (i = 0; i < n; ++i) radii[i] = +radius(nodes[i], i, nodes);
};
force.iterations = function(_) {
return arguments.length ? (iterations = +_, force) : iterations;
};
force.strength = function(_) {
return arguments.length ? (strength = +_, force) : strength;
};
force.radius = function(_) {
return arguments.length ? (radius = typeof _ === "function" ? _ : constant$6(+_), force) : radius;
};
return force;
}
function index$2(d, i) {
return i;
}
function link(links) {
var id = index$2,
strength = defaultStrength,
strengths,
distance = constant$6(30),
distances,
nodes,
count,
bias,
iterations = 1;
if (links == null) links = [];
function defaultStrength(link) {
return 1 / Math.min(count[link.source.index], count[link.target.index]);
}
function force(alpha) {
for (var k = 0, n = links.length; k < iterations; ++k) {
for (var i = 0, link, source, target, x, y, l, b; i < n; ++i) {
link = links[i], source = link.source, target = link.target;
x = target.x + target.vx - source.x - source.vx || jiggle();
y = target.y + target.vy - source.y - source.vy || jiggle();
l = Math.sqrt(x * x + y * y);
l = (l - distances[i]) / l * alpha * strengths[i];
x *= l, y *= l;
target.vx -= x * (b = bias[i]);
target.vy -= y * b;
source.vx += x * (b = 1 - b);
source.vy += y * b;
}
}
}
function initialize() {
if (!nodes) return;
var i,
n = nodes.length,
m = links.length,
nodeById = map$1(nodes, id),
link;
for (i = 0, count = new Array(n); i < n; ++i) {
count[i] = 0;
}
for (i = 0; i < m; ++i) {
link = links[i], link.index = i;
if (typeof link.source !== "object") link.source = nodeById.get(link.source);
if (typeof link.target !== "object") link.target = nodeById.get(link.target);
++count[link.source.index], ++count[link.target.index];
}
for (i = 0, bias = new Array(m); i < m; ++i) {
link = links[i], bias[i] = count[link.source.index] / (count[link.source.index] + count[link.target.index]);
}
strengths = new Array(m), initializeStrength();
distances = new Array(m), initializeDistance();
}
function initializeStrength() {
if (!nodes) return;
for (var i = 0, n = links.length; i < n; ++i) {
strengths[i] = +strength(links[i], i, links);
}
}
function initializeDistance() {
if (!nodes) return;
for (var i = 0, n = links.length; i < n; ++i) {
distances[i] = +distance(links[i], i, links);
}
}
force.initialize = function(_) {
nodes = _;
initialize();
};
force.links = function(_) {
return arguments.length ? (links = _, initialize(), force) : links;
};
force.id = function(_) {
return arguments.length ? (id = _, force) : id;
};
force.iterations = function(_) {
return arguments.length ? (iterations = +_, force) : iterations;
};
force.strength = function(_) {
return arguments.length ? (strength = typeof _ === "function" ? _ : constant$6(+_), initializeStrength(), force) : strength;
};
force.distance = function(_) {
return arguments.length ? (distance = typeof _ === "function" ? _ : constant$6(+_), initializeDistance(), force) : distance;
};
return force;
}
function x$2(d) {
return d.x;
}
function y$2(d) {
return d.y;
}
var initialRadius = 10;
var initialAngle = Math.PI * (3 - Math.sqrt(5));
function simulation(nodes) {
var simulation,
alpha = 1,
alphaMin = 0.001,
alphaDecay = 1 - Math.pow(alphaMin, 1 / 300),
alphaTarget = 0,
velocityDecay = 0.6,
forces = map$1(),
stepper = timer(step),
event = dispatch("tick", "end");
if (nodes == null) nodes = [];
function step() {
tick();
event.call("tick", simulation);
if (alpha < alphaMin) {
stepper.stop();
event.call("end", simulation);
}
}
function tick() {
var i, n = nodes.length, node;
alpha += (alphaTarget - alpha) * alphaDecay;
forces.each(function(force) {
force(alpha);
});
for (i = 0; i < n; ++i) {
node = nodes[i];
if (node.fx == null) node.x += node.vx *= velocityDecay;
else node.x = node.fx, node.vx = 0;
if (node.fy == null) node.y += node.vy *= velocityDecay;
else node.y = node.fy, node.vy = 0;
}
}
function initializeNodes() {
for (var i = 0, n = nodes.length, node; i < n; ++i) {
node = nodes[i], node.index = i;
if (isNaN(node.x) || isNaN(node.y)) {
var radius = initialRadius * Math.sqrt(i), angle = i * initialAngle;
node.x = radius * Math.cos(angle);
node.y = radius * Math.sin(angle);
}
if (isNaN(node.vx) || isNaN(node.vy)) {
node.vx = node.vy = 0;
}
}
}
function initializeForce(force) {
if (force.initialize) force.initialize(nodes);
return force;
}
initializeNodes();
return simulation = {
tick: tick,
restart: function() {
return stepper.restart(step), simulation;
},
stop: function() {
return stepper.stop(), simulation;
},
nodes: function(_) {
return arguments.length ? (nodes = _, initializeNodes(), forces.each(initializeForce), simulation) : nodes;
},
alpha: function(_) {
return arguments.length ? (alpha = +_, simulation) : alpha;
},
alphaMin: function(_) {
return arguments.length ? (alphaMin = +_, simulation) : alphaMin;
},
alphaDecay: function(_) {
return arguments.length ? (alphaDecay = +_, simulation) : +alphaDecay;
},
alphaTarget: function(_) {
return arguments.length ? (alphaTarget = +_, simulation) : alphaTarget;
},
velocityDecay: function(_) {
return arguments.length ? (velocityDecay = 1 - _, simulation) : 1 - velocityDecay;
},
force: function(name, _) {
return arguments.length > 1 ? ((_ == null ? forces.remove(name) : forces.set(name, initializeForce(_))), simulation) : forces.get(name);
},
find: function(x, y, radius) {
var i = 0,
n = nodes.length,
dx,
dy,
d2,
node,
closest;
if (radius == null) radius = Infinity;
else radius *= radius;
for (i = 0; i < n; ++i) {
node = nodes[i];
dx = x - node.x;
dy = y - node.y;
d2 = dx * dx + dy * dy;
if (d2 < radius) closest = node, radius = d2;
}
return closest;
},
on: function(name, _) {
return arguments.length > 1 ? (event.on(name, _), simulation) : event.on(name);
}
};
}
function manyBody() {
var nodes,
node,
alpha,
strength = constant$6(-30),
strengths,
distanceMin2 = 1,
distanceMax2 = Infinity,
theta2 = 0.81;
function force(_) {
var i, n = nodes.length, tree = quadtree(nodes, x$2, y$2).visitAfter(accumulate);
for (alpha = _, i = 0; i < n; ++i) node = nodes[i], tree.visit(apply);
}
function initialize() {
if (!nodes) return;
var i, n = nodes.length;
strengths = new Array(n);
for (i = 0; i < n; ++i) strengths[i] = +strength(nodes[i], i, nodes);
}
function accumulate(quad) {
var strength = 0, q, c, x, y, i;
// For internal nodes, accumulate forces from child quadrants.
if (quad.length) {
for (x = y = i = 0; i < 4; ++i) {
if ((q = quad[i]) && (c = q.value)) {
strength += c, x += c * q.x, y += c * q.y;
}
}
quad.x = x / strength;
quad.y = y / strength;
}
// For leaf nodes, accumulate forces from coincident quadrants.
else {
q = quad;
q.x = q.data.x;
q.y = q.data.y;
do strength += strengths[q.data.index];
while (q = q.next);
}
quad.value = strength;
}
function apply(quad, x1, _, x2) {
if (!quad.value) return true;
var x = quad.x - node.x,
y = quad.y - node.y,
w = x2 - x1,
l = x * x + y * y;
// Apply the Barnes-Hut approximation if possible.
// Limit forces for very close nodes; randomize direction if coincident.
if (w * w / theta2 < l) {
if (l < distanceMax2) {
if (x === 0) x = jiggle(), l += x * x;
if (y === 0) y = jiggle(), l += y * y;
if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
node.vx += x * quad.value * alpha / l;
node.vy += y * quad.value * alpha / l;
}
return true;
}
// Otherwise, process points directly.
else if (quad.length || l >= distanceMax2) return;
// Limit forces for very close nodes; randomize direction if coincident.
if (quad.data !== node || quad.next) {
if (x === 0) x = jiggle(), l += x * x;
if (y === 0) y = jiggle(), l += y * y;
if (l < distanceMin2) l = Math.sqrt(distanceMin2 * l);
}
do if (quad.data !== node) {
w = strengths[quad.data.index] * alpha / l;
node.vx += x * w;
node.vy += y * w;
} while (quad = quad.next);
}
force.initialize = function(_) {
nodes = _;
initialize();
};
force.strength = function(_) {
return arguments.length ? (strength = typeof _ === "function" ? _ : constant$6(+_), initialize(), force) : strength;
};
force.distanceMin = function(_) {
return arguments.length ? (distanceMin2 = _ * _, force) : Math.sqrt(distanceMin2);
};
force.distanceMax = function(_) {
return arguments.length ? (distanceMax2 = _ * _, force) : Math.sqrt(distanceMax2);
};
force.theta = function(_) {
return arguments.length ? (theta2 = _ * _, force) : Math.sqrt(theta2);
};
return force;
}
function x$3(x) {
var strength = constant$6(0.1),
nodes,
strengths,
xz;
if (typeof x !== "function") x = constant$6(x == null ? 0 : +x);
function force(alpha) {
for (var i = 0, n = nodes.length, node; i < n; ++i) {
node = nodes[i], node.vx += (xz[i] - node.x) * strengths[i] * alpha;
}
}
function initialize() {
if (!nodes) return;
var i, n = nodes.length;
strengths = new Array(n);
xz = new Array(n);
for (i = 0; i < n; ++i) {
strengths[i] = isNaN(xz[i] = +x(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
}
}
force.initialize = function(_) {
nodes = _;
initialize();
};
force.strength = function(_) {
return arguments.length ? (strength = typeof _ === "function" ? _ : constant$6(+_), initialize(), force) : strength;
};
force.x = function(_) {
return arguments.length ? (x = typeof _ === "function" ? _ : constant$6(+_), initialize(), force) : x;
};
return force;
}
function y$3(y) {
var strength = constant$6(0.1),
nodes,
strengths,
yz;
if (typeof y !== "function") y = constant$6(y == null ? 0 : +y);
function force(alpha) {
for (var i = 0, n = nodes.length, node; i < n; ++i) {
node = nodes[i], node.vy += (yz[i] - node.y) * strengths[i] * alpha;
}
}
function initialize() {
if (!nodes) return;
var i, n = nodes.length;
strengths = new Array(n);
yz = new Array(n);
for (i = 0; i < n; ++i) {
strengths[i] = isNaN(yz[i] = +y(nodes[i], i, nodes)) ? 0 : +strength(nodes[i], i, nodes);
}
}
force.initialize = function(_) {
nodes = _;
initialize();
};
force.strength = function(_) {
return arguments.length ? (strength = typeof _ === "function" ? _ : constant$6(+_), initialize(), force) : strength;
};
force.y = function(_) {
return arguments.length ? (y = typeof _ === "function" ? _ : constant$6(+_), initialize(), force) : y;
};
return force;
}
function nopropagation() {
exports.event.stopImmediatePropagation();
}
function noevent() {
exports.event.preventDefault();
exports.event.stopImmediatePropagation();
}
function dragDisable(view) {
var root = view.document.documentElement,
selection = select(view).on("dragstart.drag", noevent, true);
if ("onselectstart" in root) {
selection.on("selectstart.drag", noevent, true);
} else {
root.__noselect = root.style.MozUserSelect;
root.style.MozUserSelect = "none";
}
}
function dragEnable(view, noclick) {
var root = view.document.documentElement,
selection = select(view).on("dragstart.drag", null);
if (noclick) {
selection.on("click.drag", noevent, true);
setTimeout(function() { selection.on("click.drag", null); }, 0);
}
if ("onselectstart" in root) {
selection.on("selectstart.drag", null);
} else {
root.style.MozUserSelect = root.__noselect;
delete root.__noselect;
}
}
function constant$7(x) {
return function() {
return x;
};
}
function DragEvent(target, type, subject, id, active, x, y, dx, dy, dispatch) {
this.target = target;
this.type = type;
this.subject = subject;
this.identifier = id;
this.active = active;
this.x = x;
this.y = y;
this.dx = dx;
this.dy = dy;
this._ = dispatch;
}
DragEvent.prototype.on = function() {
var value = this._.on.apply(this._, arguments);
return value === this._ ? this : value;
};
// Ignore right-click, since that should open the context menu.
function defaultFilter() {
return !exports.event.button;
}
function defaultContainer() {
return this.parentNode;
}
function defaultSubject(d) {
return d == null ? {x: exports.event.x, y: exports.event.y} : d;
}
function drag() {
var filter = defaultFilter,
container = defaultContainer,
subject = defaultSubject,
gestures = {},
listeners = dispatch("start", "drag", "end"),
active = 0,
mousemoving,
touchending;
function drag(selection) {
selection
.on("mousedown.drag", mousedowned)
.on("touchstart.drag", touchstarted)
.on("touchmove.drag", touchmoved)
.on("touchend.drag touchcancel.drag", touchended)
.style("-webkit-tap-highlight-color", "rgba(0,0,0,0)");
}
function mousedowned() {
if (touchending || !filter.apply(this, arguments)) return;
var gesture = beforestart("mouse", container.apply(this, arguments), mouse, this, arguments);
if (!gesture) return;
select(exports.event.view).on("mousemove.drag", mousemoved, true).on("mouseup.drag", mouseupped, true);
dragDisable(exports.event.view);
nopropagation();
mousemoving = false;
gesture("start");
}
function mousemoved() {
noevent();
mousemoving = true;
gestures.mouse("drag");
}
function mouseupped() {
select(exports.event.view).on("mousemove.drag mouseup.drag", null);
dragEnable(exports.event.view, mousemoving);
noevent();
gestures.mouse("end");
}
function touchstarted() {
if (!filter.apply(this, arguments)) return;
var touches = exports.event.changedTouches,
c = container.apply(this, arguments),
n = touches.length, i, gesture;
for (i = 0; i < n; ++i) {
if (gesture = beforestart(touches[i].identifier, c, touch, this, arguments)) {
nopropagation();
gesture("start");
}
}
}
function touchmoved() {
var touches = exports.event.changedTouches,
n = touches.length, i, gesture;
for (i = 0; i < n; ++i) {
if (gesture = gestures[touches[i].identifier]) {
noevent();
gesture("drag");
}
}
}
function touchended() {
var touches = exports.event.changedTouches,
n = touches.length, i, gesture;
if (touchending) clearTimeout(touchending);
touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
for (i = 0; i < n; ++i) {
if (gesture = gestures[touches[i].identifier]) {
nopropagation();
gesture("end");
}
}
}
function beforestart(id, container, point, that, args) {
var p = point(container, id), s, dx, dy,
sublisteners = listeners.copy();
if (!customEvent(new DragEvent(drag, "beforestart", s, id, active, p[0], p[1], 0, 0, sublisteners), function() {
if ((exports.event.subject = s = subject.apply(that, args)) == null) return false;
dx = s.x - p[0] || 0;
dy = s.y - p[1] || 0;
return true;
})) return;
return function gesture(type) {
var p0 = p, n;
switch (type) {
case "start": gestures[id] = gesture, n = active++; break;
case "end": delete gestures[id], --active; // nobreak
case "drag": p = point(container, id), n = active; break;
}
customEvent(new DragEvent(drag, type, s, id, n, p[0] + dx, p[1] + dy, p[0] - p0[0], p[1] - p0[1], sublisteners), sublisteners.apply, sublisteners, [type, that, args]);
};
}
drag.filter = function(_) {
return arguments.length ? (filter = typeof _ === "function" ? _ : constant$7(!!_), drag) : filter;
};
drag.container = function(_) {
return arguments.length ? (container = typeof _ === "function" ? _ : constant$7(_), drag) : container;
};
drag.subject = function(_) {
return arguments.length ? (subject = typeof _ === "function" ? _ : constant$7(_), drag) : subject;
};
drag.on = function() {
var value = listeners.on.apply(listeners, arguments);
return value === listeners ? drag : value;
};
return drag;
}
function constant$8(x) {
return function() {
return x;
};
}
function x$4(d) {
return d[0];
}
function y$4(d) {
return d[1];
}
function RedBlackTree() {
this._ = null; // root node
}
function RedBlackNode(node) {
node.U = // parent node
node.C = // color - true for red, false for black
node.L = // left node
node.R = // right node
node.P = // previous node
node.N = null; // next node
}
RedBlackTree.prototype = {
constructor: RedBlackTree,
insert: function(after, node) {
var parent, grandpa, uncle;
if (after) {
node.P = after;
node.N = after.N;
if (after.N) after.N.P = node;
after.N = node;
if (after.R) {
after = after.R;
while (after.L) after = after.L;
after.L = node;
} else {
after.R = node;
}
parent = after;
} else if (this._) {
after = RedBlackFirst(this._);
node.P = null;
node.N = after;
after.P = after.L = node;
parent = after;
} else {
node.P = node.N = null;
this._ = node;
parent = null;
}
node.L = node.R = null;
node.U = parent;
node.C = true;
after = node;
while (parent && parent.C) {
grandpa = parent.U;
if (parent === grandpa.L) {
uncle = grandpa.R;
if (uncle && uncle.C) {
parent.C = uncle.C = false;
grandpa.C = true;
after = grandpa;
} else {
if (after === parent.R) {
RedBlackRotateLeft(this, parent);
after = parent;
parent = after.U;
}
parent.C = false;
grandpa.C = true;
RedBlackRotateRight(this, grandpa);
}
} else {
uncle = grandpa.L;
if (uncle && uncle.C) {
parent.C = uncle.C = false;
grandpa.C = true;
after = grandpa;
} else {
if (after === parent.L) {
RedBlackRotateRight(this, parent);
after = parent;
parent = after.U;
}
parent.C = false;
grandpa.C = true;
RedBlackRotateLeft(this, grandpa);
}
}
parent = after.U;
}
this._.C = false;
},
remove: function(node) {
if (node.N) node.N.P = node.P;
if (node.P) node.P.N = node.N;
node.N = node.P = null;
var parent = node.U,
sibling,
left = node.L,
right = node.R,
next,
red;
if (!left) next = right;
else if (!right) next = left;
else next = RedBlackFirst(right);
if (parent) {
if (parent.L === node) parent.L = next;
else parent.R = next;
} else {
this._ = next;
}
if (left && right) {
red = next.C;
next.C = node.C;
next.L = left;
left.U = next;
if (next !== right) {
parent = next.U;
next.U = node.U;
node = next.R;
parent.L = node;
next.R = right;
right.U = next;
} else {
next.U = parent;
parent = next;
node = next.R;
}
} else {
red = node.C;
node = next;
}
if (node) node.U = parent;
if (red) return;
if (node && node.C) { node.C = false; return; }
do {
if (node === this._) break;
if (node === parent.L) {
sibling = parent.R;
if (sibling.C) {
sibling.C = false;
parent.C = true;
RedBlackRotateLeft(this, parent);
sibling = parent.R;
}
if ((sibling.L && sibling.L.C)
|| (sibling.R && sibling.R.C)) {
if (!sibling.R || !sibling.R.C) {
sibling.L.C = false;
sibling.C = true;
RedBlackRotateRight(this, sibling);
sibling = parent.R;
}
sibling.C = parent.C;
parent.C = sibling.R.C = false;
RedBlackRotateLeft(this, parent);
node = this._;
break;
}
} else {
sibling = parent.L;
if (sibling.C) {
sibling.C = false;
parent.C = true;
RedBlackRotateRight(this, parent);
sibling = parent.L;
}
if ((sibling.L && sibling.L.C)
|| (sibling.R && sibling.R.C)) {
if (!sibling.L || !sibling.L.C) {
sibling.R.C = false;
sibling.C = true;
RedBlackRotateLeft(this, sibling);
sibling = parent.L;
}
sibling.C = parent.C;
parent.C = sibling.L.C = false;
RedBlackRotateRight(this, parent);
node = this._;
break;
}
}
sibling.C = true;
node = parent;
parent = parent.U;
} while (!node.C);
if (node) node.C = false;
}
};
function RedBlackRotateLeft(tree, node) {
var p = node,
q = node.R,
parent = p.U;
if (parent) {
if (parent.L === p) parent.L = q;
else parent.R = q;
} else {
tree._ = q;
}
q.U = parent;
p.U = q;
p.R = q.L;
if (p.R) p.R.U = p;
q.L = p;
}
function RedBlackRotateRight(tree, node) {
var p = node,
q = node.L,
parent = p.U;
if (parent) {
if (parent.L === p) parent.L = q;
else parent.R = q;
} else {
tree._ = q;
}
q.U = parent;
p.U = q;
p.L = q.R;
if (p.L) p.L.U = p;
q.R = p;
}
function RedBlackFirst(node) {
while (node.L) node = node.L;
return node;
}
function createEdge(left, right, v0, v1) {
var edge = [null, null],
index = edges.push(edge) - 1;
edge.left = left;
edge.right = right;
if (v0) setEdgeEnd(edge, left, right, v0);
if (v1) setEdgeEnd(edge, right, left, v1);
cells[left.index].halfedges.push(index);
cells[right.index].halfedges.push(index);
return edge;
}
function createBorderEdge(left, v0, v1) {
var edge = [v0, v1];
edge.left = left;
return edge;
}
function setEdgeEnd(edge, left, right, vertex) {
if (!edge[0] && !edge[1]) {
edge[0] = vertex;
edge.left = left;
edge.right = right;
} else if (edge.left === right) {
edge[1] = vertex;
} else {
edge[0] = vertex;
}
}
// Liang–Barsky line clipping.
function clipEdge(edge, x0, y0, x1, y1) {
var a = edge[0],
b = edge[1],
ax = a[0],
ay = a[1],
bx = b[0],
by = b[1],
t0 = 0,
t1 = 1,
dx = bx - ax,
dy = by - ay,
r;
r = x0 - ax;
if (!dx && r > 0) return;
r /= dx;
if (dx < 0) {
if (r < t0) return;
if (r < t1) t1 = r;
} else if (dx > 0) {
if (r > t1) return;
if (r > t0) t0 = r;
}
r = x1 - ax;
if (!dx && r < 0) return;
r /= dx;
if (dx < 0) {
if (r > t1) return;
if (r > t0) t0 = r;
} else if (dx > 0) {
if (r < t0) return;
if (r < t1) t1 = r;
}
r = y0 - ay;
if (!dy && r > 0) return;
r /= dy;
if (dy < 0) {
if (r < t0) return;
if (r < t1) t1 = r;
} else if (dy > 0) {
if (r > t1) return;
if (r > t0) t0 = r;
}
r = y1 - ay;
if (!dy && r < 0) return;
r /= dy;
if (dy < 0) {
if (r > t1) return;
if (r > t0) t0 = r;
} else if (dy > 0) {
if (r < t0) return;
if (r < t1) t1 = r;
}
if (!(t0 > 0) && !(t1 < 1)) return true; // TODO Better check?
if (t0 > 0) edge[0] = [ax + t0 * dx, ay + t0 * dy];
if (t1 < 1) edge[1] = [ax + t1 * dx, ay + t1 * dy];
return true;
}
function connectEdge(edge, x0, y0, x1, y1) {
var v1 = edge[1];
if (v1) return true;
var v0 = edge[0],
left = edge.left,
right = edge.right,
lx = left[0],
ly = left[1],
rx = right[0],
ry = right[1],
fx = (lx + rx) / 2,
fy = (ly + ry) / 2,
fm,
fb;
if (ry === ly) {
if (fx < x0 || fx >= x1) return;
if (lx > rx) {
if (!v0) v0 = [fx, y0];
else if (v0[1] >= y1) return;
v1 = [fx, y1];
} else {
if (!v0) v0 = [fx, y1];
else if (v0[1] < y0) return;
v1 = [fx, y0];
}
} else {
fm = (lx - rx) / (ry - ly);
fb = fy - fm * fx;
if (fm < -1 || fm > 1) {
if (lx > rx) {
if (!v0) v0 = [(y0 - fb) / fm, y0];
else if (v0[1] >= y1) return;
v1 = [(y1 - fb) / fm, y1];
} else {
if (!v0) v0 = [(y1 - fb) / fm, y1];
else if (v0[1] < y0) return;
v1 = [(y0 - fb) / fm, y0];
}
} else {
if (ly < ry) {
if (!v0) v0 = [x0, fm * x0 + fb];
else if (v0[0] >= x1) return;
v1 = [x1, fm * x1 + fb];
} else {
if (!v0) v0 = [x1, fm * x1 + fb];
else if (v0[0] < x0) return;
v1 = [x0, fm * x0 + fb];
}
}
}
edge[0] = v0;
edge[1] = v1;
return true;
}
function clipEdges(x0, y0, x1, y1) {
var i = edges.length,
edge;
while (i--) {
if (!connectEdge(edge = edges[i], x0, y0, x1, y1)
|| !clipEdge(edge, x0, y0, x1, y1)
|| !(Math.abs(edge[0][0] - edge[1][0]) > epsilon$3
|| Math.abs(edge[0][1] - edge[1][1]) > epsilon$3)) {
delete edges[i];
}
}
}
function createCell(site) {
return cells[site.index] = {
site: site,
halfedges: []
};
}
function cellHalfedgeAngle(cell, edge) {
var site = cell.site,
va = edge.left,
vb = edge.right;
if (site === vb) vb = va, va = site;
if (vb) return Math.atan2(vb[1] - va[1], vb[0] - va[0]);
if (site === va) va = edge[1], vb = edge[0];
else va = edge[0], vb = edge[1];
return Math.atan2(va[0] - vb[0], vb[1] - va[1]);
}
function cellHalfedgeStart(cell, edge) {
return edge[+(edge.left !== cell.site)];
}
function cellHalfedgeEnd(cell, edge) {
return edge[+(edge.left === cell.site)];
}
function sortCellHalfedges() {
for (var i = 0, n = cells.length, cell, halfedges, j, m; i < n; ++i) {
if ((cell = cells[i]) && (m = (halfedges = cell.halfedges).length)) {
var index = new Array(m),
array = new Array(m);
for (j = 0; j < m; ++j) index[j] = j, array[j] = cellHalfedgeAngle(cell, edges[halfedges[j]]);
index.sort(function(i, j) { return array[j] - array[i]; });
for (j = 0; j < m; ++j) array[j] = halfedges[index[j]];
for (j = 0; j < m; ++j) halfedges[j] = array[j];
}
}
}
function clipCells(x0, y0, x1, y1) {
var nCells = cells.length,
iCell,
cell,
site,
iHalfedge,
halfedges,
nHalfedges,
start,
startX,
startY,
end,
endX,
endY,
cover = true;
for (iCell = 0; iCell < nCells; ++iCell) {
if (cell = cells[iCell]) {
site = cell.site;
halfedges = cell.halfedges;
iHalfedge = halfedges.length;
// Remove any dangling clipped edges.
while (iHalfedge--) {
if (!edges[halfedges[iHalfedge]]) {
halfedges.splice(iHalfedge, 1);
}
}
// Insert any border edges as necessary.
iHalfedge = 0, nHalfedges = halfedges.length;
while (iHalfedge < nHalfedges) {
end = cellHalfedgeEnd(cell, edges[halfedges[iHalfedge]]), endX = end[0], endY = end[1];
start = cellHalfedgeStart(cell, edges[halfedges[++iHalfedge % nHalfedges]]), startX = start[0], startY = start[1];
if (Math.abs(endX - startX) > epsilon$3 || Math.abs(endY - startY) > epsilon$3) {
halfedges.splice(iHalfedge, 0, edges.push(createBorderEdge(site, end,
Math.abs(endX - x0) < epsilon$3 && y1 - endY > epsilon$3 ? [x0, Math.abs(startX - x0) < epsilon$3 ? startY : y1]
: Math.abs(endY - y1) < epsilon$3 && x1 - endX > epsilon$3 ? [Math.abs(startY - y1) < epsilon$3 ? startX : x1, y1]
: Math.abs(endX - x1) < epsilon$3 && endY - y0 > epsilon$3 ? [x1, Math.abs(startX - x1) < epsilon$3 ? startY : y0]
: Math.abs(endY - y0) < epsilon$3 && endX - x0 > epsilon$3 ? [Math.abs(startY - y0) < epsilon$3 ? startX : x0, y0]
: null)) - 1);
++nHalfedges;
}
}
if (nHalfedges) cover = false;
}
}
// If there weren’t any edges, have the closest site cover the extent.
// It doesn’t matter which corner of the extent we measure!
if (cover) {
var dx, dy, d2, dc = Infinity;
for (iCell = 0, cover = null; iCell < nCells; ++iCell) {
if (cell = cells[iCell]) {
site = cell.site;
dx = site[0] - x0;
dy = site[1] - y0;
d2 = dx * dx + dy * dy;
if (d2 < dc) dc = d2, cover = cell;
}
}
if (cover) {
var v00 = [x0, y0], v01 = [x0, y1], v11 = [x1, y1], v10 = [x1, y0];
cover.halfedges.push(
edges.push(createBorderEdge(site = cover.site, v00, v01)) - 1,
edges.push(createBorderEdge(site, v01, v11)) - 1,
edges.push(createBorderEdge(site, v11, v10)) - 1,
edges.push(createBorderEdge(site, v10, v00)) - 1
);
}
}
// Lastly delete any cells with no edges; these were entirely clipped.
for (iCell = 0; iCell < nCells; ++iCell) {
if (cell = cells[iCell]) {
if (!cell.halfedges.length) {
delete cells[iCell];
}
}
}
}
var circlePool = [];
var firstCircle;
function Circle() {
RedBlackNode(this);
this.x =
this.y =
this.arc =
this.site =
this.cy = null;
}
function attachCircle(arc) {
var lArc = arc.P,
rArc = arc.N;
if (!lArc || !rArc) return;
var lSite = lArc.site,
cSite = arc.site,
rSite = rArc.site;
if (lSite === rSite) return;
var bx = cSite[0],
by = cSite[1],
ax = lSite[0] - bx,
ay = lSite[1] - by,
cx = rSite[0] - bx,
cy = rSite[1] - by;
var d = 2 * (ax * cy - ay * cx);
if (d >= -epsilon2$1) return;
var ha = ax * ax + ay * ay,
hc = cx * cx + cy * cy,
x = (cy * ha - ay * hc) / d,
y = (ax * hc - cx * ha) / d;
var circle = circlePool.pop() || new Circle;
circle.arc = arc;
circle.site = cSite;
circle.x = x + bx;
circle.y = (circle.cy = y + by) + Math.sqrt(x * x + y * y); // y bottom
arc.circle = circle;
var before = null,
node = circles._;
while (node) {
if (circle.y < node.y || (circle.y === node.y && circle.x <= node.x)) {
if (node.L) node = node.L;
else { before = node.P; break; }
} else {
if (node.R) node = node.R;
else { before = node; break; }
}
}
circles.insert(before, circle);
if (!before) firstCircle = circle;
}
function detachCircle(arc) {
var circle = arc.circle;
if (circle) {
if (!circle.P) firstCircle = circle.N;
circles.remove(circle);
circlePool.push(circle);
RedBlackNode(circle);
arc.circle = null;
}
}
var beachPool = [];
function Beach() {
RedBlackNode(this);
this.edge =
this.site =
this.circle = null;
}
function createBeach(site) {
var beach = beachPool.pop() || new Beach;
beach.site = site;
return beach;
}
function detachBeach(beach) {
detachCircle(beach);
beaches.remove(beach);
beachPool.push(beach);
RedBlackNode(beach);
}
function removeBeach(beach) {
var circle = beach.circle,
x = circle.x,
y = circle.cy,
vertex = [x, y],
previous = beach.P,
next = beach.N,
disappearing = [beach];
detachBeach(beach);
var lArc = previous;
while (lArc.circle
&& Math.abs(x - lArc.circle.x) < epsilon$3
&& Math.abs(y - lArc.circle.cy) < epsilon$3) {
previous = lArc.P;
disappearing.unshift(lArc);
detachBeach(lArc);
lArc = previous;
}
disappearing.unshift(lArc);
detachCircle(lArc);
var rArc = next;
while (rArc.circle
&& Math.abs(x - rArc.circle.x) < epsilon$3
&& Math.abs(y - rArc.circle.cy) < epsilon$3) {
next = rArc.N;
disappearing.push(rArc);
detachBeach(rArc);
rArc = next;
}
disappearing.push(rArc);
detachCircle(rArc);
var nArcs = disappearing.length,
iArc;
for (iArc = 1; iArc < nArcs; ++iArc) {
rArc = disappearing[iArc];
lArc = disappearing[iArc - 1];
setEdgeEnd(rArc.edge, lArc.site, rArc.site, vertex);
}
lArc = disappearing[0];
rArc = disappearing[nArcs - 1];
rArc.edge = createEdge(lArc.site, rArc.site, null, vertex);
attachCircle(lArc);
attachCircle(rArc);
}
function addBeach(site) {
var x = site[0],
directrix = site[1],
lArc,
rArc,
dxl,
dxr,
node = beaches._;
while (node) {
dxl = leftBreakPoint(node, directrix) - x;
if (dxl > epsilon$3) node = node.L; else {
dxr = x - rightBreakPoint(node, directrix);
if (dxr > epsilon$3) {
if (!node.R) {
lArc = node;
break;
}
node = node.R;
} else {
if (dxl > -epsilon$3) {
lArc = node.P;
rArc = node;
} else if (dxr > -epsilon$3) {
lArc = node;
rArc = node.N;
} else {
lArc = rArc = node;
}
break;
}
}
}
createCell(site);
var newArc = createBeach(site);
beaches.insert(lArc, newArc);
if (!lArc && !rArc) return;
if (lArc === rArc) {
detachCircle(lArc);
rArc = createBeach(lArc.site);
beaches.insert(newArc, rArc);
newArc.edge = rArc.edge = createEdge(lArc.site, newArc.site);
attachCircle(lArc);
attachCircle(rArc);
return;
}
if (!rArc) { // && lArc
newArc.edge = createEdge(lArc.site, newArc.site);
return;
}
// else lArc !== rArc
detachCircle(lArc);
detachCircle(rArc);
var lSite = lArc.site,
ax = lSite[0],
ay = lSite[1],
bx = site[0] - ax,
by = site[1] - ay,
rSite = rArc.site,
cx = rSite[0] - ax,
cy = rSite[1] - ay,
d = 2 * (bx * cy - by * cx),
hb = bx * bx + by * by,
hc = cx * cx + cy * cy,
vertex = [(cy * hb - by * hc) / d + ax, (bx * hc - cx * hb) / d + ay];
setEdgeEnd(rArc.edge, lSite, rSite, vertex);
newArc.edge = createEdge(lSite, site, null, vertex);
rArc.edge = createEdge(site, rSite, null, vertex);
attachCircle(lArc);
attachCircle(rArc);
}
function leftBreakPoint(arc, directrix) {
var site = arc.site,
rfocx = site[0],
rfocy = site[1],
pby2 = rfocy - directrix;
if (!pby2) return rfocx;
var lArc = arc.P;
if (!lArc) return -Infinity;
site = lArc.site;
var lfocx = site[0],
lfocy = site[1],
plby2 = lfocy - directrix;
if (!plby2) return lfocx;
var hl = lfocx - rfocx,
aby2 = 1 / pby2 - 1 / plby2,
b = hl / plby2;
if (aby2) return (-b + Math.sqrt(b * b - 2 * aby2 * (hl * hl / (-2 * plby2) - lfocy + plby2 / 2 + rfocy - pby2 / 2))) / aby2 + rfocx;
return (rfocx + lfocx) / 2;
}
function rightBreakPoint(arc, directrix) {
var rArc = arc.N;
if (rArc) return leftBreakPoint(rArc, directrix);
var site = arc.site;
return site[1] === directrix ? site[0] : Infinity;
}
var epsilon$3 = 1e-6;
var epsilon2$1 = 1e-12;
var beaches;
var cells;
var circles;
var edges;
function triangleArea(a, b, c) {
return (a[0] - c[0]) * (b[1] - a[1]) - (a[0] - b[0]) * (c[1] - a[1]);
}
function lexicographic(a, b) {
return b[1] - a[1]
|| b[0] - a[0];
}
function Diagram(sites, extent) {
var site = sites.sort(lexicographic).pop(),
x,
y,
circle;
edges = [];
cells = new Array(sites.length);
beaches = new RedBlackTree;
circles = new RedBlackTree;
while (true) {
circle = firstCircle;
if (site && (!circle || site[1] < circle.y || (site[1] === circle.y && site[0] < circle.x))) {
if (site[0] !== x || site[1] !== y) {
addBeach(site);
x = site[0], y = site[1];
}
site = sites.pop();
} else if (circle) {
removeBeach(circle.arc);
} else {
break;
}
}
sortCellHalfedges();
if (extent) {
var x0 = +extent[0][0],
y0 = +extent[0][1],
x1 = +extent[1][0],
y1 = +extent[1][1];
clipEdges(x0, y0, x1, y1);
clipCells(x0, y0, x1, y1);
}
this.edges = edges;
this.cells = cells;
beaches =
circles =
edges =
cells = null;
}
Diagram.prototype = {
constructor: Diagram,
polygons: function() {
var edges = this.edges;
return this.cells.map(function(cell) {
var polygon = cell.halfedges.map(function(i) { return cellHalfedgeStart(cell, edges[i]); });
polygon.data = cell.site.data;
return polygon;
});
},
triangles: function() {
var triangles = [],
edges = this.edges;
this.cells.forEach(function(cell, i) {
var site = cell.site,
halfedges = cell.halfedges,
j = -1,
m = halfedges.length,
s0,
e1 = edges[halfedges[m - 1]],
s1 = e1.left === site ? e1.right : e1.left;
while (++j < m) {
s0 = s1;
e1 = edges[halfedges[j]];
s1 = e1.left === site ? e1.right : e1.left;
if (i < s0.index && i < s1.index && triangleArea(site, s0, s1) < 0) {
triangles.push([site.data, s0.data, s1.data]);
}
}
});
return triangles;
},
links: function() {
return this.edges.filter(function(edge) {
return edge.right;
}).map(function(edge) {
return {
source: edge.left.data,
target: edge.right.data
};
});
}
}
function voronoi() {
var x = x$4,
y = y$4,
extent = null;
function voronoi(data) {
return new Diagram(data.map(function(d, i) {
var s = [Math.round(x(d, i, data) / epsilon$3) * epsilon$3, Math.round(y(d, i, data) / epsilon$3) * epsilon$3];
s.index = i;
s.data = d;
return s;
}), extent);
}
voronoi.polygons = function(data) {
return voronoi(data).polygons();
};
voronoi.links = function(data) {
return voronoi(data).links();
};
voronoi.triangles = function(data) {
return voronoi(data).triangles();
};
voronoi.x = function(_) {
return arguments.length ? (x = typeof _ === "function" ? _ : constant$8(+_), voronoi) : x;
};
voronoi.y = function(_) {
return arguments.length ? (y = typeof _ === "function" ? _ : constant$8(+_), voronoi) : y;
};
voronoi.extent = function(_) {
return arguments.length ? (extent = _ == null ? null : [[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]], voronoi) : extent && [[extent[0][0], extent[0][1]], [extent[1][0], extent[1][1]]];
};
voronoi.size = function(_) {
return arguments.length ? (extent = _ == null ? null : [[0, 0], [+_[0], +_[1]]], voronoi) : extent && [extent[1][0] - extent[0][0], extent[1][1] - extent[0][1]];
};
return voronoi;
}
function constant$9(x) {
return function() {
return x;
};
}
function ZoomEvent(target, type, transform) {
this.target = target;
this.type = type;
this.transform = transform;
}
function Transform(k, x, y) {
this.k = k;
this.x = x;
this.y = y;
}
Transform.prototype = {
constructor: Transform,
scale: function(k) {
return k === 1 ? this : new Transform(this.k * k, this.x, this.y);
},
translate: function(x, y) {
return x === 0 & y === 0 ? this : new Transform(this.k, this.x + this.k * x, this.y + this.k * y);
},
apply: function(point) {
return [point[0] * this.k + this.x, point[1] * this.k + this.y];
},
applyX: function(x) {
return x * this.k + this.x;
},
applyY: function(y) {
return y * this.k + this.y;
},
invert: function(location) {
return [(location[0] - this.x) / this.k, (location[1] - this.y) / this.k];
},
invertX: function(x) {
return (x - this.x) / this.k;
},
invertY: function(y) {
return (y - this.y) / this.k;
},
rescaleX: function(x) {
return x.copy().domain(x.range().map(this.invertX, this).map(x.invert, x));
},
rescaleY: function(y) {
return y.copy().domain(y.range().map(this.invertY, this).map(y.invert, y));
},
toString: function() {
return "translate(" + this.x + "," + this.y + ") scale(" + this.k + ")";
}
};
var identity$6 = new Transform(1, 0, 0);
transform.prototype = Transform.prototype;
function transform(node) {
return node.__zoom || identity$6;
}
function nopropagation$1() {
exports.event.stopImmediatePropagation();
}
function noevent$1() {
exports.event.preventDefault();
exports.event.stopImmediatePropagation();
}
// Ignore right-click, since that should open the context menu.
function defaultFilter$1() {
return !exports.event.button;
}
function defaultExtent() {
var e = this, w, h;
if (e instanceof SVGElement) {
e = e.ownerSVGElement || e;
w = e.width.baseVal.value;
h = e.height.baseVal.value;
} else {
w = e.clientWidth;
h = e.clientHeight;
}
return [[0, 0], [w, h]];
}
function defaultTransform() {
return this.__zoom || identity$6;
}
function zoom() {
var filter = defaultFilter$1,
extent = defaultExtent,
k0 = 0,
k1 = Infinity,
x0 = -k1,
x1 = k1,
y0 = x0,
y1 = x1,
duration = 250,
gestures = [],
listeners = dispatch("start", "zoom", "end"),
touchstarting,
touchending,
touchDelay = 500,
wheelDelay = 150;
function zoom(selection) {
selection
.on("wheel.zoom", wheeled)
.on("mousedown.zoom", mousedowned)
.on("dblclick.zoom", dblclicked)
.on("touchstart.zoom", touchstarted)
.on("touchmove.zoom", touchmoved)
.on("touchend.zoom touchcancel.zoom", touchended)
.style("-webkit-tap-highlight-color", "rgba(0,0,0,0)")
.property("__zoom", defaultTransform);
}
zoom.transform = function(collection, transform) {
var selection = collection.selection ? collection.selection() : collection;
selection.property("__zoom", defaultTransform);
if (collection !== selection) {
schedule(collection, transform);
} else {
selection.interrupt().each(function() {
gesture(this, arguments)
.start()
.zoom(null, typeof transform === "function" ? transform.apply(this, arguments) : transform)
.end();
});
}
};
zoom.scaleBy = function(selection, k) {
zoom.scaleTo(selection, function() {
var k0 = this.__zoom.k,
k1 = typeof k === "function" ? k.apply(this, arguments) : k;
return k0 * k1;
});
};
zoom.scaleTo = function(selection, k) {
zoom.transform(selection, function() {
var e = extent.apply(this, arguments),
t0 = this.__zoom,
p0 = centroid(e),
p1 = t0.invert(p0),
k1 = typeof k === "function" ? k.apply(this, arguments) : k;
return constrain(translate(scale(t0, k1), p0, p1), e);
});
};
zoom.translateBy = function(selection, x, y) {
zoom.transform(selection, function() {
return constrain(this.__zoom.translate(
typeof x === "function" ? x.apply(this, arguments) : x,
typeof y === "function" ? y.apply(this, arguments) : y
), extent.apply(this, arguments));
});
};
function scale(transform, k) {
k = Math.max(k0, Math.min(k1, k));
return k === transform.k ? transform : new Transform(k, transform.x, transform.y);
}
function translate(transform, p0, p1) {
var x = p0[0] - p1[0] * transform.k, y = p0[1] - p1[1] * transform.k;
return x === transform.x && y === transform.y ? transform : new Transform(transform.k, x, y);
}
function constrain(transform, extent) {
var dx = Math.min(0, transform.invertX(extent[0][0]) - x0) || Math.max(0, transform.invertX(extent[1][0]) - x1),
dy = Math.min(0, transform.invertY(extent[0][1]) - y0) || Math.max(0, transform.invertY(extent[1][1]) - y1);
return dx || dy ? transform.translate(dx, dy) : transform;
}
function centroid(extent) {
return [(+extent[0][0] + +extent[1][0]) / 2, (+extent[0][1] + +extent[1][1]) / 2];
}
function schedule(transition, transform, center) {
transition
.on("start.zoom", function() { gesture(this, arguments).start(); })
.on("interrupt.zoom end.zoom", function() { gesture(this, arguments).end(); })
.tween("zoom", function() {
var that = this,
args = arguments,
g = gesture(that, args),
e = extent.apply(that, args),
p = center || centroid(e),
w = Math.max(e[1][0] - e[0][0], e[1][1] - e[0][1]),
a = that.__zoom,
b = typeof transform === "function" ? transform.apply(that, args) : transform,
i = interpolateZoom(a.invert(p).concat(w / a.k), b.invert(p).concat(w / b.k));
return function(t) {
if (t === 1) t = b; // Avoid rounding error on end.
else { var l = i(t), k = w / l[2]; t = new Transform(k, p[0] - l[0] * k, p[1] - l[1] * k); }
g.zoom(null, t);
};
});
}
function gesture(that, args) {
for (var i = 0, n = gestures.length, g; i < n; ++i) {
if ((g = gestures[i]).that === that) {
return g;
}
}
return new Gesture(that, args);
}
function Gesture(that, args) {
this.that = that;
this.args = args;
this.index = -1;
this.active = 0;
this.extent = extent.apply(that, args);
}
Gesture.prototype = {
start: function() {
if (++this.active === 1) {
this.index = gestures.push(this) - 1;
this.emit("start");
}
return this;
},
zoom: function(key, transform) {
if (this.mouse && key !== "mouse") this.mouse[1] = transform.invert(this.mouse[0]);
if (this.touch0 && key !== "touch") this.touch0[1] = transform.invert(this.touch0[0]);
if (this.touch1 && key !== "touch") this.touch1[1] = transform.invert(this.touch1[0]);
this.that.__zoom = transform;
this.emit("zoom");
return this;
},
end: function() {
if (--this.active === 0) {
gestures.splice(this.index, 1);
this.index = -1;
this.emit("end");
}
return this;
},
emit: function(type) {
customEvent(new ZoomEvent(zoom, type, this.that.__zoom), listeners.apply, listeners, [type, this.that, this.args]);
}
};
function wheeled() {
if (!filter.apply(this, arguments)) return;
var g = gesture(this, arguments),
t = this.__zoom,
k = Math.max(k0, Math.min(k1, t.k * Math.pow(2, -exports.event.deltaY * (exports.event.deltaMode ? 120 : 1) / 500))),
p = mouse(this);
// If the mouse is in the same location as before, reuse it.
// If there were recent wheel events, reset the wheel idle timeout.
if (g.wheel) {
if (g.mouse[0][0] !== p[0] || g.mouse[0][1] !== p[1]) {
g.mouse[1] = t.invert(g.mouse[0] = p);
}
clearTimeout(g.wheel);
}
// If this wheel event won’t trigger a transform change, ignore it.
else if (t.k === k) return;
// Otherwise, capture the mouse point and location at the start.
else {
g.mouse = [p, t.invert(p)];
interrupt(this);
g.start();
}
noevent$1();
g.wheel = setTimeout(wheelidled, wheelDelay);
g.zoom("mouse", constrain(translate(scale(t, k), g.mouse[0], g.mouse[1]), g.extent));
function wheelidled() {
g.wheel = null;
g.end();
}
}
function mousedowned() {
if (touchending || !filter.apply(this, arguments)) return;
var g = gesture(this, arguments),
v = select(exports.event.view).on("mousemove.zoom", mousemoved, true).on("mouseup.zoom", mouseupped, true),
p = mouse(this);
dragDisable(exports.event.view);
nopropagation$1();
g.mouse = [p, this.__zoom.invert(p)];
interrupt(this);
g.start();
function mousemoved() {
noevent$1();
g.moved = true;
g.zoom("mouse", constrain(translate(g.that.__zoom, g.mouse[0] = mouse(g.that), g.mouse[1]), g.extent));
}
function mouseupped() {
v.on("mousemove.zoom mouseup.zoom", null);
dragEnable(exports.event.view, g.moved);
noevent$1();
g.end();
}
}
function dblclicked() {
if (!filter.apply(this, arguments)) return;
var t0 = this.__zoom,
p0 = mouse(this),
p1 = t0.invert(p0),
k1 = t0.k * (exports.event.shiftKey ? 0.5 : 2),
t1 = constrain(translate(scale(t0, k1), p0, p1), extent.apply(this, arguments));
noevent$1();
if (duration > 0) select(this).transition().duration(duration).call(schedule, t1, p0);
else select(this).call(zoom.transform, t1);
}
function touchstarted() {
if (!filter.apply(this, arguments)) return;
var g = gesture(this, arguments),
touches = exports.event.changedTouches,
n = touches.length, i, t, p;
nopropagation$1();
for (i = 0; i < n; ++i) {
t = touches[i], p = touch(this, touches, t.identifier);
p = [p, this.__zoom.invert(p), t.identifier];
if (!g.touch0) g.touch0 = p;
else if (!g.touch1) g.touch1 = p;
}
if (touchstarting) {
touchstarting = clearTimeout(touchstarting);
if (!g.touch1) return g.end(), dblclicked.apply(this, arguments);
}
if (exports.event.touches.length === n) {
touchstarting = setTimeout(function() { touchstarting = null; }, touchDelay);
interrupt(this);
g.start();
}
}
function touchmoved() {
var g = gesture(this, arguments),
touches = exports.event.changedTouches,
n = touches.length, i, t, p, l;
noevent$1();
if (touchstarting) touchstarting = clearTimeout(touchstarting);
for (i = 0; i < n; ++i) {
t = touches[i], p = touch(this, touches, t.identifier);
if (g.touch0 && g.touch0[2] === t.identifier) g.touch0[0] = p;
else if (g.touch1 && g.touch1[2] === t.identifier) g.touch1[0] = p;
}
t = g.that.__zoom;
if (g.touch1) {
var p0 = g.touch0[0], l0 = g.touch0[1],
p1 = g.touch1[0], l1 = g.touch1[1],
dp = (dp = p1[0] - p0[0]) * dp + (dp = p1[1] - p0[1]) * dp,
dl = (dl = l1[0] - l0[0]) * dl + (dl = l1[1] - l0[1]) * dl;
t = scale(t, Math.sqrt(dp / dl));
p = [(p0[0] + p1[0]) / 2, (p0[1] + p1[1]) / 2];
l = [(l0[0] + l1[0]) / 2, (l0[1] + l1[1]) / 2];
}
else if (g.touch0) p = g.touch0[0], l = g.touch0[1];
else return;
g.zoom("touch", constrain(translate(t, p, l), g.extent));
}
function touchended() {
var g = gesture(this, arguments),
touches = exports.event.changedTouches,
n = touches.length, i, t;
nopropagation$1();
if (touchending) clearTimeout(touchending);
touchending = setTimeout(function() { touchending = null; }, touchDelay);
for (i = 0; i < n; ++i) {
t = touches[i];
if (g.touch0 && g.touch0[2] === t.identifier) delete g.touch0;
else if (g.touch1 && g.touch1[2] === t.identifier) delete g.touch1;
}
if (g.touch1 && !g.touch0) g.touch0 = g.touch1, delete g.touch1;
if (!g.touch0) g.end();
}
zoom.filter = function(_) {
return arguments.length ? (filter = typeof _ === "function" ? _ : constant$9(!!_), zoom) : filter;
};
zoom.extent = function(_) {
return arguments.length ? (extent = typeof _ === "function" ? _ : constant$9([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), zoom) : extent;
};
zoom.scaleExtent = function(_) {
return arguments.length ? (k0 = +_[0], k1 = +_[1], zoom) : [k0, k1];
};
zoom.translateExtent = function(_) {
return arguments.length ? (x0 = +_[0][0], x1 = +_[1][0], y0 = +_[0][1], y1 = +_[1][1], zoom) : [[x0, y0], [x1, y1]];
};
zoom.duration = function(_) {
return arguments.length ? (duration = +_, zoom) : duration;
};
zoom.on = function() {
var value = listeners.on.apply(listeners, arguments);
return value === listeners ? zoom : value;
};
return zoom;
}
function constant$10(x) {
return function() {
return x;
};
}
function BrushEvent(target, type, selection) {
this.target = target;
this.type = type;
this.selection = selection;
}
function nopropagation$2() {
exports.event.stopImmediatePropagation();
}
function noevent$2() {
exports.event.preventDefault();
exports.event.stopImmediatePropagation();
}
var MODE_DRAG = {name: "drag"};
var MODE_SPACE = {name: "space"};
var MODE_HANDLE = {name: "handle"};
var MODE_CENTER = {name: "center"};
var X = {
name: "x",
handles: ["e", "w"].map(type$1),
input: function(x, e) { return x && [[x[0], e[0][1]], [x[1], e[1][1]]]; },
output: function(xy) { return xy && [xy[0][0], xy[1][0]]; }
};
var Y = {
name: "y",
handles: ["n", "s"].map(type$1),
input: function(y, e) { return y && [[e[0][0], y[0]], [e[1][0], y[1]]]; },
output: function(xy) { return xy && [xy[0][1], xy[1][1]]; }
};
var XY = {
name: "xy",
handles: ["n", "e", "s", "w", "nw", "ne", "se", "sw"].map(type$1),
input: function(xy) { return xy; },
output: function(xy) { return xy; }
};
var cursors = {
overlay: "crosshair",
selection: "move",
n: "ns-resize",
e: "ew-resize",
s: "ns-resize",
w: "ew-resize",
nw: "nwse-resize",
ne: "nesw-resize",
se: "nwse-resize",
sw: "nesw-resize"
};
var flipX = {
e: "w",
w: "e",
nw: "ne",
ne: "nw",
se: "sw",
sw: "se"
};
var flipY = {
n: "s",
s: "n",
nw: "sw",
ne: "se",
se: "ne",
sw: "nw"
};
var signsX = {
overlay: +1,
selection: +1,
n: null,
e: +1,
s: null,
w: -1,
nw: -1,
ne: +1,
se: +1,
sw: -1
};
var signsY = {
overlay: +1,
selection: +1,
n: -1,
e: null,
s: +1,
w: null,
nw: -1,
ne: -1,
se: +1,
sw: +1
};
function type$1(t) {
return {type: t};
}
// Ignore right-click, since that should open the context menu.
function defaultFilter$2() {
return !exports.event.button;
}
function defaultExtent$1() {
var svg = this.ownerSVGElement || this;
return [[0, 0], [svg.width.baseVal.value, svg.height.baseVal.value]];
}
// Like d3.local, but with the name “__brush” rather than auto-generated.
function local$1(node) {
while (!node.__brush) if (!(node = node.parentNode)) return;
return node.__brush;
}
function empty$1(extent) {
return extent[0][0] === extent[1][0]
|| extent[0][1] === extent[1][1];
}
function brushSelection(node) {
var state = node.__brush;
return state ? state.dim.output(state.selection) : null;
}
function brushX() {
return brush$1(X);
}
function brushY() {
return brush$1(Y);
}
function brush() {
return brush$1(XY);
}
function brush$1(dim) {
var extent = defaultExtent$1,
filter = defaultFilter$2,
listeners = dispatch(brush, "start", "brush", "end"),
handleSize = 6,
touchending;
function brush(group) {
var overlay = group
.property("__brush", initialize)
.selectAll(".overlay")
.data([type$1("overlay")]);
overlay.enter().append("rect")
.attr("class", "overlay")
.attr("pointer-events", "all")
.attr("cursor", cursors.overlay)
.merge(overlay)
.each(function() {
var extent = local$1(this).extent;
select(this)
.attr("x", extent[0][0])
.attr("y", extent[0][1])
.attr("width", extent[1][0] - extent[0][0])
.attr("height", extent[1][1] - extent[0][1]);
});
group.selectAll(".selection")
.data([type$1("selection")])
.enter().append("rect")
.attr("class", "selection")
.attr("cursor", cursors.selection)
.attr("fill", "#777")
.attr("fill-opacity", 0.3)
.attr("stroke", "#fff")
.attr("shape-rendering", "crispEdges");
var handle = group.selectAll(".handle")
.data(dim.handles, function(d) { return d.type; });
handle.exit().remove();
handle.enter().append("rect")
.attr("class", function(d) { return "handle handle--" + d.type; })
.attr("cursor", function(d) { return cursors[d.type]; });
group
.each(redraw)
.attr("fill", "none")
.attr("pointer-events", "all")
.style("-webkit-tap-highlight-color", "rgba(0,0,0,0)")
.on("mousedown.brush touchstart.brush", started);
}
brush.move = function(group, selection) {
if (group.selection) {
group
.on("start.brush", function() { emitter(this, arguments).beforestart().start(); })
.on("interrupt.brush end.brush", function() { emitter(this, arguments).end(); })
.tween("brush", function() {
var that = this,
state = that.__brush,
emit = emitter(that, arguments),
selection0 = state.selection,
selection1 = dim.input(typeof selection === "function" ? selection.apply(this, arguments) : selection, state.extent),
i = interpolate(selection0, selection1);
function tween(t) {
state.selection = t === 1 && empty$1(selection1) ? null : i(t);
redraw.call(that);
emit.brush();
}
return selection0 && selection1 ? tween : tween(1);
});
} else {
group
.each(function() {
var that = this,
args = arguments,
state = that.__brush,
selection1 = dim.input(typeof selection === "function" ? selection.apply(that, args) : selection, state.extent),
emit = emitter(that, args).beforestart();
interrupt(that);
state.selection = selection1 == null || empty$1(selection1) ? null : selection1;
redraw.call(that);
emit.start().brush().end();
});
}
};
function redraw() {
var group = select(this),
selection = local$1(this).selection;
if (selection) {
group.selectAll(".selection")
.style("display", null)
.attr("x", selection[0][0])
.attr("y", selection[0][1])
.attr("width", selection[1][0] - selection[0][0])
.attr("height", selection[1][1] - selection[0][1]);
group.selectAll(".handle")
.style("display", null)
.attr("x", function(d) { return d.type[d.type.length - 1] === "e" ? selection[1][0] - handleSize / 2 : selection[0][0] - handleSize / 2; })
.attr("y", function(d) { return d.type[0] === "s" ? selection[1][1] - handleSize / 2 : selection[0][1] - handleSize / 2; })
.attr("width", function(d) { return d.type === "n" || d.type === "s" ? selection[1][0] - selection[0][0] + handleSize : handleSize; })
.attr("height", function(d) { return d.type === "e" || d.type === "w" ? selection[1][1] - selection[0][1] + handleSize : handleSize; });
}
else {
group.selectAll(".selection,.handle")
.style("display", "none")
.attr("x", null)
.attr("y", null)
.attr("width", null)
.attr("height", null);
}
}
function emitter(that, args) {
return that.__brush.emitter || new Emitter(that, args);
}
function Emitter(that, args) {
this.that = that;
this.args = args;
this.state = that.__brush;
this.active = 0;
}
Emitter.prototype = {
beforestart: function() {
if (++this.active === 1) this.state.emitter = this, this.starting = true;
return this;
},
start: function() {
if (this.starting) this.starting = false, this.emit("start");
return this;
},
brush: function() {
this.emit("brush");
return this;
},
end: function() {
if (--this.active === 0) delete this.state.emitter, this.emit("end");
return this;
},
emit: function(type) {
customEvent(new BrushEvent(brush, type, dim.output(this.state.selection)), listeners.apply, listeners, [type, this.that, this.args]);
}
};
function started() {
if (exports.event.touches) { if (exports.event.changedTouches.length < exports.event.touches.length) return noevent$2(); }
else if (touchending) return;
if (!filter.apply(this, arguments)) return;
var that = this,
type = exports.event.target.__data__.type,
mode = (exports.event.metaKey ? type = "overlay" : type) === "selection" ? MODE_DRAG : (exports.event.altKey ? MODE_CENTER : MODE_HANDLE),
signX = dim === Y ? null : signsX[type],
signY = dim === X ? null : signsY[type],
state = local$1(that),
extent = state.extent,
selection = state.selection,
W = extent[0][0], w0, w1,
N = extent[0][1], n0, n1,
E = extent[1][0], e0, e1,
S = extent[1][1], s0, s1,
dx,
dy,
moving,
shifting = signX && signY && exports.event.shiftKey,
lockX,
lockY,
point0 = mouse(that),
point = point0,
emit = emitter(that, arguments).beforestart();
if (type === "overlay") {
state.selection = selection = [
[w0 = dim === Y ? W : point0[0], n0 = dim === X ? N : point0[1]],
[e0 = dim === Y ? E : w0, s0 = dim === X ? S : n0]
];
} else {
w0 = selection[0][0];
n0 = selection[0][1];
e0 = selection[1][0];
s0 = selection[1][1];
}
w1 = w0;
n1 = n0;
e1 = e0;
s1 = s0;
var group = select(that)
.attr("pointer-events", "none");
var overlay = group.selectAll(".overlay")
.attr("cursor", cursors[type]);
if (exports.event.touches) {
group
.on("touchmove.brush", moved, true)
.on("touchend.brush touchcancel.brush", ended, true);
} else {
var view = select(exports.event.view)
.on("keydown.brush", keydowned, true)
.on("keyup.brush", keyupped, true)
.on("mousemove.brush", moved, true)
.on("mouseup.brush", ended, true);
dragDisable(exports.event.view);
}
nopropagation$2();
interrupt(that);
redraw.call(that);
emit.start();
function moved() {
var point1 = mouse(that);
if (shifting && !lockX && !lockY) {
if (Math.abs(point1[0] - point[0]) > Math.abs(point1[1] - point[1])) lockY = true;
else lockX = true;
}
point = point1;
moving = true;
noevent$2();
move();
}
function move() {
var t;
dx = point[0] - point0[0];
dy = point[1] - point0[1];
switch (mode) {
case MODE_SPACE:
case MODE_DRAG: {
if (signX) dx = Math.max(W - w0, Math.min(E - e0, dx)), w1 = w0 + dx, e1 = e0 + dx;
if (signY) dy = Math.max(N - n0, Math.min(S - s0, dy)), n1 = n0 + dy, s1 = s0 + dy;
break;
}
case MODE_HANDLE: {
if (signX < 0) dx = Math.max(W - w0, Math.min(E - w0, dx)), w1 = w0 + dx, e1 = e0;
else if (signX > 0) dx = Math.max(W - e0, Math.min(E - e0, dx)), w1 = w0, e1 = e0 + dx;
if (signY < 0) dy = Math.max(N - n0, Math.min(S - n0, dy)), n1 = n0 + dy, s1 = s0;
else if (signY > 0) dy = Math.max(N - s0, Math.min(S - s0, dy)), n1 = n0, s1 = s0 + dy;
break;
}
case MODE_CENTER: {
if (signX) w1 = Math.max(W, Math.min(E, w0 - dx * signX)), e1 = Math.max(W, Math.min(E, e0 + dx * signX));
if (signY) n1 = Math.max(N, Math.min(S, n0 - dy * signY)), s1 = Math.max(N, Math.min(S, s0 + dy * signY));
break;
}
}
if (e1 < w1) {
signX *= -1;
t = w0, w0 = e0, e0 = t;
t = w1, w1 = e1, e1 = t;
if (type in flipX) overlay.attr("cursor", cursors[type = flipX[type]]);
}
if (s1 < n1) {
signY *= -1;
t = n0, n0 = s0, s0 = t;
t = n1, n1 = s1, s1 = t;
if (type in flipY) overlay.attr("cursor", cursors[type = flipY[type]]);
}
selection = state.selection; // May be set by brush.move!
if (lockX) w1 = selection[0][0], e1 = selection[1][0];
if (lockY) n1 = selection[0][1], s1 = selection[1][1];
if (selection[0][0] !== w1
|| selection[0][1] !== n1
|| selection[1][0] !== e1
|| selection[1][1] !== s1) {
state.selection = [[w1, n1], [e1, s1]];
redraw.call(that);
emit.brush();
}
}
function ended() {
nopropagation$2();
if (exports.event.touches) {
if (exports.event.touches.length) return;
if (touchending) clearTimeout(touchending);
touchending = setTimeout(function() { touchending = null; }, 500); // Ghost clicks are delayed!
group.on("touchmove.brush touchend.brush touchcancel.brush", null);
} else {
dragEnable(exports.event.view, moving);
view.on("keydown.brush keyup.brush mousemove.brush mouseup.brush", null);
}
group.attr("pointer-events", "all");
overlay.attr("cursor", cursors.overlay);
if (empty$1(selection)) state.selection = null, redraw.call(that);
emit.end();
}
function keydowned() {
switch (exports.event.keyCode) {
case 16: { // SHIFT
shifting = signX && signY;
break;
}
case 18: { // ALT
if (mode === MODE_HANDLE) {
if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
mode = MODE_CENTER;
move();
}
break;
}
case 32: { // SPACE; takes priority over ALT
if (mode === MODE_HANDLE || mode === MODE_CENTER) {
if (signX < 0) e0 = e1 - dx; else if (signX > 0) w0 = w1 - dx;
if (signY < 0) s0 = s1 - dy; else if (signY > 0) n0 = n1 - dy;
mode = MODE_SPACE;
overlay.attr("cursor", cursors.selection);
move();
}
break;
}
default: return;
}
noevent$2();
}
function keyupped() {
switch (exports.event.keyCode) {
case 16: { // SHIFT
if (shifting) {
lockX = lockY = shifting = false;
move();
}
break;
}
case 18: { // ALT
if (mode === MODE_CENTER) {
if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
mode = MODE_HANDLE;
move();
}
break;
}
case 32: { // SPACE
if (mode === MODE_SPACE) {
if (exports.event.altKey) {
if (signX) e0 = e1 - dx * signX, w0 = w1 + dx * signX;
if (signY) s0 = s1 - dy * signY, n0 = n1 + dy * signY;
mode = MODE_CENTER;
} else {
if (signX < 0) e0 = e1; else if (signX > 0) w0 = w1;
if (signY < 0) s0 = s1; else if (signY > 0) n0 = n1;
mode = MODE_HANDLE;
}
overlay.attr("cursor", cursors[type]);
move();
}
break;
}
default: return;
}
noevent$2();
}
}
function initialize() {
var state = this.__brush || {selection: null};
state.extent = extent.apply(this, arguments);
state.dim = dim;
return state;
}
brush.extent = function(_) {
return arguments.length ? (extent = typeof _ === "function" ? _ : constant$10([[+_[0][0], +_[0][1]], [+_[1][0], +_[1][1]]]), brush) : extent;
};
brush.filter = function(_) {
return arguments.length ? (filter = typeof _ === "function" ? _ : constant$10(!!_), brush) : filter;
};
brush.handleSize = function(_) {
return arguments.length ? (handleSize = +_, brush) : handleSize;
};
brush.on = function() {
var value = listeners.on.apply(listeners, arguments);
return value === listeners ? brush : value;
};
return brush;
}
// Adds floating point numbers with twice the normal precision.
// Reference: J. R. Shewchuk, Adaptive Precision Floating-Point Arithmetic and
// Fast Robust Geometric Predicates, Discrete & Computational Geometry 18(3)
// 305–363 (1997).
// Code adapted from GeographicLib by Charles F. F. Karney,
// http://geographiclib.sourceforge.net/
function adder() {
return new Adder;
}
function Adder() {
this.reset();
}
Adder.prototype = {
constructor: Adder,
reset: function() {
this.s = // rounded value
this.t = 0; // exact error
},
add: function(y) {
add$1(temp, y, this.t);
add$1(this, temp.s, this.s);
if (this.s) this.t += temp.t;
else this.s = temp.t;
},
valueOf: function() {
return this.s;
}
};
var temp = new Adder;
function add$1(adder, a, b) {
var x = adder.s = a + b,
bv = x - a,
av = x - bv;
adder.t = (a - av) + (b - bv);
}
var epsilon$4 = 1e-6;
var epsilon2$2 = 1e-12;
var pi$3 = Math.PI;
var halfPi$2 = pi$3 / 2;
var quarterPi = pi$3 / 4;
var tau$3 = pi$3 * 2;
var degrees$1 = 180 / pi$3;
var radians = pi$3 / 180;
var abs = Math.abs;
var atan = Math.atan;
var atan2 = Math.atan2;
var cos = Math.cos;
var ceil = Math.ceil;
var exp = Math.exp;
var log$1 = Math.log;
var pow$1 = Math.pow;
var sin = Math.sin;
var sign$1 = Math.sign || function(x) { return x > 0 ? 1 : x < 0 ? -1 : 0; };
var sqrt$1 = Math.sqrt;
var tan = Math.tan;
function acos(x) {
return x > 1 ? 0 : x < -1 ? pi$3 : Math.acos(x);
}
function asin$1(x) {
return x > 1 ? halfPi$2 : x < -1 ? -halfPi$2 : Math.asin(x);
}
function haversin(x) {
return (x = sin(x / 2)) * x;
}
function noop$2() {}
function streamGeometry(geometry, stream) {
if (geometry && streamGeometryType.hasOwnProperty(geometry.type)) {
streamGeometryType[geometry.type](geometry, stream);
}
}
var streamObjectType = {
Feature: function(feature, stream) {
streamGeometry(feature.geometry, stream);
},
FeatureCollection: function(object, stream) {
var features = object.features, i = -1, n = features.length;
while (++i < n) streamGeometry(features[i].geometry, stream);
}
};
var streamGeometryType = {
Sphere: function(object, stream) {
stream.sphere();
},
Point: function(object, stream) {
object = object.coordinates;
stream.point(object[0], object[1], object[2]);
},
MultiPoint: function(object, stream) {
var coordinates = object.coordinates, i = -1, n = coordinates.length;
while (++i < n) object = coordinates[i], stream.point(object[0], object[1], object[2]);
},
LineString: function(object, stream) {
streamLine(object.coordinates, stream, 0);
},
MultiLineString: function(object, stream) {
var coordinates = object.coordinates, i = -1, n = coordinates.length;
while (++i < n) streamLine(coordinates[i], stream, 0);
},
Polygon: function(object, stream) {
streamPolygon(object.coordinates, stream);
},
MultiPolygon: function(object, stream) {
var coordinates = object.coordinates, i = -1, n = coordinates.length;
while (++i < n) streamPolygon(coordinates[i], stream);
},
GeometryCollection: function(object, stream) {
var geometries = object.geometries, i = -1, n = geometries.length;
while (++i < n) streamGeometry(geometries[i], stream);
}
};
function streamLine(coordinates, stream, closed) {
var i = -1, n = coordinates.length - closed, coordinate;
stream.lineStart();
while (++i < n) coordinate = coordinates[i], stream.point(coordinate[0], coordinate[1], coordinate[2]);
stream.lineEnd();
}
function streamPolygon(coordinates, stream) {
var i = -1, n = coordinates.length;
stream.polygonStart();
while (++i < n) streamLine(coordinates[i], stream, 1);
stream.polygonEnd();
}
function geoStream(object, stream) {
if (object && streamObjectType.hasOwnProperty(object.type)) {
streamObjectType[object.type](object, stream);
} else {
streamGeometry(object, stream);
}
}
var areaRingSum = adder();
var areaSum = adder();
var lambda00;
var phi00;
var lambda0;
var cosPhi0;
var sinPhi0;
var areaStream = {
point: noop$2,
lineStart: noop$2,
lineEnd: noop$2,
polygonStart: function() {
areaRingSum.reset();
areaStream.lineStart = areaRingStart;
areaStream.lineEnd = areaRingEnd;
},
polygonEnd: function() {
var areaRing = +areaRingSum;
areaSum.add(areaRing < 0 ? tau$3 + areaRing : areaRing);
this.lineStart = this.lineEnd = this.point = noop$2;
},
sphere: function() {
areaSum.add(tau$3);
}
};
function areaRingStart() {
areaStream.point = areaPointFirst;
}
function areaRingEnd() {
areaPoint(lambda00, phi00);
}
function areaPointFirst(lambda, phi) {
areaStream.point = areaPoint;
lambda00 = lambda, phi00 = phi;
lambda *= radians, phi *= radians;
lambda0 = lambda, cosPhi0 = cos(phi = phi / 2 + quarterPi), sinPhi0 = sin(phi);
}
function areaPoint(lambda, phi) {
lambda *= radians, phi *= radians;
phi = phi / 2 + quarterPi; // half the angular distance from south pole
// Spherical excess E for a spherical triangle with vertices: south pole,
// previous point, current point. Uses a formula derived from Cagnoli’s
// theorem. See Todhunter, Spherical Trig. (1871), Sec. 103, Eq. (2).
var dLambda = lambda - lambda0,
sdLambda = dLambda >= 0 ? 1 : -1,
adLambda = sdLambda * dLambda,
cosPhi = cos(phi),
sinPhi = sin(phi),
k = sinPhi0 * sinPhi,
u = cosPhi0 * cosPhi + k * cos(adLambda),
v = k * sdLambda * sin(adLambda);
areaRingSum.add(atan2(v, u));
// Advance the previous points.
lambda0 = lambda, cosPhi0 = cosPhi, sinPhi0 = sinPhi;
}
function area$2(object) {
areaSum.reset();
geoStream(object, areaStream);
return areaSum * 2;
}
function spherical(cartesian) {
return [atan2(cartesian[1], cartesian[0]), asin$1(cartesian[2])];
}
function cartesian(spherical) {
var lambda = spherical[0], phi = spherical[1], cosPhi = cos(phi);
return [cosPhi * cos(lambda), cosPhi * sin(lambda), sin(phi)];
}
function cartesianDot(a, b) {
return a[0] * b[0] + a[1] * b[1] + a[2] * b[2];
}
function cartesianCross(a, b) {
return [a[1] * b[2] - a[2] * b[1], a[2] * b[0] - a[0] * b[2], a[0] * b[1] - a[1] * b[0]];
}
// TODO return a
function cartesianAddInPlace(a, b) {
a[0] += b[0], a[1] += b[1], a[2] += b[2];
}
function cartesianScale(vector, k) {
return [vector[0] * k, vector[1] * k, vector[2] * k];
}
// TODO return d
function cartesianNormalizeInPlace(d) {
var l = sqrt$1(d[0] * d[0] + d[1] * d[1] + d[2] * d[2]);
d[0] /= l, d[1] /= l, d[2] /= l;
}
var lambda0$1;
var phi0;
var lambda1;
var phi1;
var lambda2;
var lambda00$1;
var phi00$1;
var p0;
var deltaSum = adder();
var ranges;
var range$1;
var boundsStream = {
point: boundsPoint,
lineStart: boundsLineStart,
lineEnd: boundsLineEnd,
polygonStart: function() {
boundsStream.point = boundsRingPoint;
boundsStream.lineStart = boundsRingStart;
boundsStream.lineEnd = boundsRingEnd;
deltaSum.reset();
areaStream.polygonStart();
},
polygonEnd: function() {
areaStream.polygonEnd();
boundsStream.point = boundsPoint;
boundsStream.lineStart = boundsLineStart;
boundsStream.lineEnd = boundsLineEnd;
if (areaRingSum < 0) lambda0$1 = -(lambda1 = 180), phi0 = -(phi1 = 90);
else if (deltaSum > epsilon$4) phi1 = 90;
else if (deltaSum < -epsilon$4) phi0 = -90;
range$1[0] = lambda0$1, range$1[1] = lambda1;
}
};
function boundsPoint(lambda, phi) {
ranges.push(range$1 = [lambda0$1 = lambda, lambda1 = lambda]);
if (phi < phi0) phi0 = phi;
if (phi > phi1) phi1 = phi;
}
function linePoint(lambda, phi) {
var p = cartesian([lambda * radians, phi * radians]);
if (p0) {
var normal = cartesianCross(p0, p),
equatorial = [normal[1], -normal[0], 0],
inflection = cartesianCross(equatorial, normal);
cartesianNormalizeInPlace(inflection);
inflection = spherical(inflection);
var delta = lambda - lambda2,
sign = delta > 0 ? 1 : -1,
lambdai = inflection[0] * degrees$1 * sign,
phii,
antimeridian = abs(delta) > 180;
if (antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
phii = inflection[1] * degrees$1;
if (phii > phi1) phi1 = phii;
} else if (lambdai = (lambdai + 360) % 360 - 180, antimeridian ^ (sign * lambda2 < lambdai && lambdai < sign * lambda)) {
phii = -inflection[1] * degrees$1;
if (phii < phi0) phi0 = phii;
} else {
if (phi < phi0) phi0 = phi;
if (phi > phi1) phi1 = phi;
}
if (antimeridian) {
if (lambda < lambda2) {
if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
} else {
if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
}
} else {
if (lambda1 >= lambda0$1) {
if (lambda < lambda0$1) lambda0$1 = lambda;
if (lambda > lambda1) lambda1 = lambda;
} else {
if (lambda > lambda2) {
if (angle(lambda0$1, lambda) > angle(lambda0$1, lambda1)) lambda1 = lambda;
} else {
if (angle(lambda, lambda1) > angle(lambda0$1, lambda1)) lambda0$1 = lambda;
}
}
}
} else {
boundsPoint(lambda, phi);
}
p0 = p, lambda2 = lambda;
}
function boundsLineStart() {
boundsStream.point = linePoint;
}
function boundsLineEnd() {
range$1[0] = lambda0$1, range$1[1] = lambda1;
boundsStream.point = boundsPoint;
p0 = null;
}
function boundsRingPoint(lambda, phi) {
if (p0) {
var delta = lambda - lambda2;
deltaSum.add(abs(delta) > 180 ? delta + (delta > 0 ? 360 : -360) : delta);
} else {
lambda00$1 = lambda, phi00$1 = phi;
}
areaStream.point(lambda, phi);
linePoint(lambda, phi);
}
function boundsRingStart() {
areaStream.lineStart();
}
function boundsRingEnd() {
boundsRingPoint(lambda00$1, phi00$1);
areaStream.lineEnd();
if (abs(deltaSum) > epsilon$4) lambda0$1 = -(lambda1 = 180);
range$1[0] = lambda0$1, range$1[1] = lambda1;
p0 = null;
}
// Finds the left-right distance between two longitudes.
// This is almost the same as (lambda1 - lambda0 + 360°) % 360°, except that we want
// the distance between ±180° to be 360°.
function angle(lambda0, lambda1) {
return (lambda1 -= lambda0) < 0 ? lambda1 + 360 : lambda1;
}
function rangeCompare(a, b) {
return a[0] - b[0];
}
function rangeContains(range, x) {
return range[0] <= range[1] ? range[0] <= x && x <= range[1] : x < range[0] || range[1] < x;
}
function bounds(feature) {
var i, n, a, b, merged, deltaMax, delta;
phi1 = lambda1 = -(lambda0$1 = phi0 = Infinity);
ranges = [];
geoStream(feature, boundsStream);
// First, sort ranges by their minimum longitudes.
if (n = ranges.length) {
ranges.sort(rangeCompare);
// Then, merge any ranges that overlap.
for (i = 1, a = ranges[0], merged = [a]; i < n; ++i) {
b = ranges[i];
if (rangeContains(a, b[0]) || rangeContains(a, b[1])) {
if (angle(a[0], b[1]) > angle(a[0], a[1])) a[1] = b[1];
if (angle(b[0], a[1]) > angle(a[0], a[1])) a[0] = b[0];
} else {
merged.push(a = b);
}
}
// Finally, find the largest gap between the merged ranges.
// The final bounding box will be the inverse of this gap.
for (deltaMax = -Infinity, n = merged.length - 1, i = 0, a = merged[n]; i <= n; a = b, ++i) {
b = merged[i];
if ((delta = angle(a[1], b[0])) > deltaMax) deltaMax = delta, lambda0$1 = b[0], lambda1 = a[1];
}
}
ranges = range$1 = null;
return lambda0$1 === Infinity || phi0 === Infinity
? [[NaN, NaN], [NaN, NaN]]
: [[lambda0$1, phi0], [lambda1, phi1]];
}
var W0;
var W1;
var X0;
var Y0;
var Z0;
var X1;
var Y1;
var Z1;
var X2;
var Y2;
var Z2;
var lambda00$2;
var phi00$2;
var x0;
var y0;
var z0;
// previous point
var centroidStream = {
sphere: noop$2,
point: centroidPoint,
lineStart: centroidLineStart,
lineEnd: centroidLineEnd,
polygonStart: function() {
centroidStream.lineStart = centroidRingStart;
centroidStream.lineEnd = centroidRingEnd;
},
polygonEnd: function() {
centroidStream.lineStart = centroidLineStart;
centroidStream.lineEnd = centroidLineEnd;
}
};
// Arithmetic mean of Cartesian vectors.
function centroidPoint(lambda, phi) {
lambda *= radians, phi *= radians;
var cosPhi = cos(phi);
centroidPointCartesian(cosPhi * cos(lambda), cosPhi * sin(lambda), sin(phi));
}
function centroidPointCartesian(x, y, z) {
++W0;
X0 += (x - X0) / W0;
Y0 += (y - Y0) / W0;
Z0 += (z - Z0) / W0;
}
function centroidLineStart() {
centroidStream.point = centroidLinePointFirst;
}
function centroidLinePointFirst(lambda, phi) {
lambda *= radians, phi *= radians;
var cosPhi = cos(phi);
x0 = cosPhi * cos(lambda);
y0 = cosPhi * sin(lambda);
z0 = sin(phi);
centroidStream.point = centroidLinePoint;
centroidPointCartesian(x0, y0, z0);
}
function centroidLinePoint(lambda, phi) {
lambda *= radians, phi *= radians;
var cosPhi = cos(phi),
x = cosPhi * cos(lambda),
y = cosPhi * sin(lambda),
z = sin(phi),
w = atan2(sqrt$1((w = y0 * z - z0 * y) * w + (w = z0 * x - x0 * z) * w + (w = x0 * y - y0 * x) * w), x0 * x + y0 * y + z0 * z);
W1 += w;
X1 += w * (x0 + (x0 = x));
Y1 += w * (y0 + (y0 = y));
Z1 += w * (z0 + (z0 = z));
centroidPointCartesian(x0, y0, z0);
}
function centroidLineEnd() {
centroidStream.point = centroidPoint;
}
// See J. E. Brock, The Inertia Tensor for a Spherical Triangle,
// J. Applied Mechanics 42, 239 (1975).
function centroidRingStart() {
centroidStream.point = centroidRingPointFirst;
}
function centroidRingEnd() {
centroidRingPoint(lambda00$2, phi00$2);
centroidStream.point = centroidPoint;
}
function centroidRingPointFirst(lambda, phi) {
lambda00$2 = lambda, phi00$2 = phi;
lambda *= radians, phi *= radians;
centroidStream.point = centroidRingPoint;
var cosPhi = cos(phi);
x0 = cosPhi * cos(lambda);
y0 = cosPhi * sin(lambda);
z0 = sin(phi);
centroidPointCartesian(x0, y0, z0);
}
function centroidRingPoint(lambda, phi) {
lambda *= radians, phi *= radians;
var cosPhi = cos(phi),
x = cosPhi * cos(lambda),
y = cosPhi * sin(lambda),
z = sin(phi),
cx = y0 * z - z0 * y,
cy = z0 * x - x0 * z,
cz = x0 * y - y0 * x,
m = sqrt$1(cx * cx + cy * cy + cz * cz),
u = x0 * x + y0 * y + z0 * z,
v = m && -acos(u) / m, // area weight
w = atan2(m, u); // line weight
X2 += v * cx;
Y2 += v * cy;
Z2 += v * cz;
W1 += w;
X1 += w * (x0 + (x0 = x));
Y1 += w * (y0 + (y0 = y));
Z1 += w * (z0 + (z0 = z));
centroidPointCartesian(x0, y0, z0);
}
function centroid$1(object) {
W0 = W1 =
X0 = Y0 = Z0 =
X1 = Y1 = Z1 =
X2 = Y2 = Z2 = 0;
geoStream(object, centroidStream);
var x = X2,
y = Y2,
z = Z2,
m = x * x + y * y + z * z;
// If the area-weighted ccentroid is undefined, fall back to length-weighted ccentroid.
if (m < epsilon2$2) {
x = X1, y = Y1, z = Z1;
// If the feature has zero length, fall back to arithmetic mean of point vectors.
if (W1 < epsilon$4) x = X0, y = Y0, z = Z0;
m = x * x + y * y + z * z;
// If the feature still has an undefined ccentroid, then return.
if (m < epsilon2$2) return [NaN, NaN];
}
return [atan2(y, x) * degrees$1, asin$1(z / sqrt$1(m)) * degrees$1];
}
function constant$11(x) {
return function() {
return x;
};
}
function compose(a, b) {
function compose(x, y) {
return x = a(x, y), b(x[0], x[1]);
}
if (a.invert && b.invert) compose.invert = function(x, y) {
return x = b.invert(x, y), x && a.invert(x[0], x[1]);
};
return compose;
}
function rotationIdentity(lambda, phi) {
return [lambda > pi$3 ? lambda - tau$3 : lambda < -pi$3 ? lambda + tau$3 : lambda, phi];
}
rotationIdentity.invert = rotationIdentity;
function rotateRadians(deltaLambda, deltaPhi, deltaGamma) {
return (deltaLambda %= tau$3) ? (deltaPhi || deltaGamma ? compose(rotationLambda(deltaLambda), rotationPhiGamma(deltaPhi, deltaGamma))
: rotationLambda(deltaLambda))
: (deltaPhi || deltaGamma ? rotationPhiGamma(deltaPhi, deltaGamma)
: rotationIdentity);
}
function forwardRotationLambda(deltaLambda) {
return function(lambda, phi) {
return lambda += deltaLambda, [lambda > pi$3 ? lambda - tau$3 : lambda < -pi$3 ? lambda + tau$3 : lambda, phi];
};
}
function rotationLambda(deltaLambda) {
var rotation = forwardRotationLambda(deltaLambda);
rotation.invert = forwardRotationLambda(-deltaLambda);
return rotation;
}
function rotationPhiGamma(deltaPhi, deltaGamma) {
var cosDeltaPhi = cos(deltaPhi),
sinDeltaPhi = sin(deltaPhi),
cosDeltaGamma = cos(deltaGamma),
sinDeltaGamma = sin(deltaGamma);
function rotation(lambda, phi) {
var cosPhi = cos(phi),
x = cos(lambda) * cosPhi,
y = sin(lambda) * cosPhi,
z = sin(phi),
k = z * cosDeltaPhi + x * sinDeltaPhi;
return [
atan2(y * cosDeltaGamma - k * sinDeltaGamma, x * cosDeltaPhi - z * sinDeltaPhi),
asin$1(k * cosDeltaGamma + y * sinDeltaGamma)
];
}
rotation.invert = function(lambda, phi) {
var cosPhi = cos(phi),
x = cos(lambda) * cosPhi,
y = sin(lambda) * cosPhi,
z = sin(phi),
k = z * cosDeltaGamma - y * sinDeltaGamma;
return [
atan2(y * cosDeltaGamma + z * sinDeltaGamma, x * cosDeltaPhi + k * sinDeltaPhi),
asin$1(k * cosDeltaPhi - x * sinDeltaPhi)
];
};
return rotation;
}
function rotation(rotate) {
rotate = rotateRadians(rotate[0] * radians, rotate[1] * radians, rotate.length > 2 ? rotate[2] * radians : 0);
function forward(coordinates) {
coordinates = rotate(coordinates[0] * radians, coordinates[1] * radians);
return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
}
forward.invert = function(coordinates) {
coordinates = rotate.invert(coordinates[0] * radians, coordinates[1] * radians);
return coordinates[0] *= degrees$1, coordinates[1] *= degrees$1, coordinates;
};
return forward;
}
// Generates a circle centered at [0°, 0°], with a given radius and precision.
function circleStream(stream, radius, delta, direction, t0, t1) {
if (!delta) return;
var cosRadius = cos(radius),
sinRadius = sin(radius),
step = direction * delta;
if (t0 == null) {
t0 = radius + direction * tau$3;
t1 = radius - step / 2;
} else {
t0 = circleRadius(cosRadius, t0);
t1 = circleRadius(cosRadius, t1);
if (direction > 0 ? t0 < t1 : t0 > t1) t0 += direction * tau$3;
}
for (var point, t = t0; direction > 0 ? t > t1 : t < t1; t -= step) {
point = spherical([cosRadius, -sinRadius * cos(t), -sinRadius * sin(t)]);
stream.point(point[0], point[1]);
}
}
// Returns the signed angle of a cartesian point relative to [cosRadius, 0, 0].
function circleRadius(cosRadius, point) {
point = cartesian(point), point[0] -= cosRadius;
cartesianNormalizeInPlace(point);
var radius = acos(-point[1]);
return ((-point[2] < 0 ? -radius : radius) + tau$3 - epsilon$4) % tau$3;
}
function circle$1() {
var center = constant$11([0, 0]),
radius = constant$11(90),
precision = constant$11(6),
ring,
rotate,
stream = {point: point};
function point(x, y) {
ring.push(x = rotate(x, y));
x[0] *= degrees$1, x[1] *= degrees$1;
}
function circle() {
var c = center.apply(this, arguments),
r = radius.apply(this, arguments) * radians,
p = precision.apply(this, arguments) * radians;
ring = [];
rotate = rotateRadians(-c[0] * radians, -c[1] * radians, 0).invert;
circleStream(stream, r, p, 1);
c = {type: "Polygon", coordinates: [ring]};
ring = rotate = null;
return c;
}
circle.center = function(_) {
return arguments.length ? (center = typeof _ === "function" ? _ : constant$11([+_[0], +_[1]]), circle) : center;
};
circle.radius = function(_) {
return arguments.length ? (radius = typeof _ === "function" ? _ : constant$11(+_), circle) : radius;
};
circle.precision = function(_) {
return arguments.length ? (precision = typeof _ === "function" ? _ : constant$11(+_), circle) : precision;
};
return circle;
}
function clipBuffer() {
var lines = [],
line;
return {
point: function(x, y) {
line.push([x, y]);
},
lineStart: function() {
lines.push(line = []);
},
lineEnd: noop$2,
rejoin: function() {
if (lines.length > 1) lines.push(lines.pop().concat(lines.shift()));
},
result: function() {
var result = lines;
lines = [];
line = null;
return result;
}
};
}
function clipLine(a, b, x0, y0, x1, y1) {
var ax = a[0],
ay = a[1],
bx = b[0],
by = b[1],
t0 = 0,
t1 = 1,
dx = bx - ax,
dy = by - ay,
r;
r = x0 - ax;
if (!dx && r > 0) return;
r /= dx;
if (dx < 0) {
if (r < t0) return;
if (r < t1) t1 = r;
} else if (dx > 0) {
if (r > t1) return;
if (r > t0) t0 = r;
}
r = x1 - ax;
if (!dx && r < 0) return;
r /= dx;
if (dx < 0) {
if (r > t1) return;
if (r > t0) t0 = r;
} else if (dx > 0) {
if (r < t0) return;
if (r < t1) t1 = r;
}
r = y0 - ay;
if (!dy && r > 0) return;
r /= dy;
if (dy < 0) {
if (r < t0) return;
if (r < t1) t1 = r;
} else if (dy > 0) {
if (r > t1) return;
if (r > t0) t0 = r;
}
r = y1 - ay;
if (!dy && r < 0) return;
r /= dy;
if (dy < 0) {
if (r > t1) return;
if (r > t0) t0 = r;
} else if (dy > 0) {
if (r < t0) return;
if (r < t1) t1 = r;
}
if (t0 > 0) a[0] = ax + t0 * dx, a[1] = ay + t0 * dy;
if (t1 < 1) b[0] = ax + t1 * dx, b[1] = ay + t1 * dy;
return true;
}
function pointEqual(a, b) {
return abs(a[0] - b[0]) < epsilon$4 && abs(a[1] - b[1]) < epsilon$4;
}
function Intersection(point, points, other, entry) {
this.x = point;
this.z = points;
this.o = other; // another intersection
this.e = entry; // is an entry?
this.v = false; // visited
this.n = this.p = null; // next & previous
}
// A generalized polygon clipping algorithm: given a polygon that has been cut
// into its visible line segments, and rejoins the segments by interpolating
// along the clip edge.
function clipPolygon(segments, compareIntersection, startInside, interpolate, stream) {
var subject = [],
clip = [],
i,
n;
segments.forEach(function(segment) {
if ((n = segment.length - 1) <= 0) return;
var n, p0 = segment[0], p1 = segment[n], x;
// If the first and last points of a segment are coincident, then treat as a
// closed ring. TODO if all rings are closed, then the winding order of the
// exterior ring should be checked.
if (pointEqual(p0, p1)) {
stream.lineStart();
for (i = 0; i < n; ++i) stream.point((p0 = segment[i])[0], p0[1]);
stream.lineEnd();
return;
}
subject.push(x = new Intersection(p0, segment, null, true));
clip.push(x.o = new Intersection(p0, null, x, false));
subject.push(x = new Intersection(p1, segment, null, false));
clip.push(x.o = new Intersection(p1, null, x, true));
});
if (!subject.length) return;
clip.sort(compareIntersection);
link$1(subject);
link$1(clip);
for (i = 0, n = clip.length; i < n; ++i) {
clip[i].e = startInside = !startInside;
}
var start = subject[0],
points,
point;
while (1) {
// Find first unvisited intersection.
var current = start,
isSubject = true;
while (current.v) if ((current = current.n) === start) return;
points = current.z;
stream.lineStart();
do {
current.v = current.o.v = true;
if (current.e) {
if (isSubject) {
for (i = 0, n = points.length; i < n; ++i) stream.point((point = points[i])[0], point[1]);
} else {
interpolate(current.x, current.n.x, 1, stream);
}
current = current.n;
} else {
if (isSubject) {
points = current.p.z;
for (i = points.length - 1; i >= 0; --i) stream.point((point = points[i])[0], point[1]);
} else {
interpolate(current.x, current.p.x, -1, stream);
}
current = current.p;
}
current = current.o;
points = current.z;
isSubject = !isSubject;
} while (!current.v);
stream.lineEnd();
}
}
function link$1(array) {
if (!(n = array.length)) return;
var n,
i = 0,
a = array[0],
b;
while (++i < n) {
a.n = b = array[i];
b.p = a;
a = b;
}
a.n = b = array[0];
b.p = a;
}
var clipMax = 1e9;
var clipMin = -clipMax;
// TODO Use d3-polygon’s polygonContains here for the ring check?
// TODO Eliminate duplicate buffering in clipBuffer and polygon.push?
function clipExtent(x0, y0, x1, y1) {
function visible(x, y) {
return x0 <= x && x <= x1 && y0 <= y && y <= y1;
}
function interpolate(from, to, direction, stream) {
var a = 0, a1 = 0;
if (from == null
|| (a = corner(from, direction)) !== (a1 = corner(to, direction))
|| comparePoint(from, to) < 0 ^ direction > 0) {
do stream.point(a === 0 || a === 3 ? x0 : x1, a > 1 ? y1 : y0);
while ((a = (a + direction + 4) % 4) !== a1);
} else {
stream.point(to[0], to[1]);
}
}
function corner(p, direction) {
return abs(p[0] - x0) < epsilon$4 ? direction > 0 ? 0 : 3
: abs(p[0] - x1) < epsilon$4 ? direction > 0 ? 2 : 1
: abs(p[1] - y0) < epsilon$4 ? direction > 0 ? 1 : 0
: direction > 0 ? 3 : 2; // abs(p[1] - y1) < epsilon
}
function compareIntersection(a, b) {
return comparePoint(a.x, b.x);
}
function comparePoint(a, b) {
var ca = corner(a, 1),
cb = corner(b, 1);
return ca !== cb ? ca - cb
: ca === 0 ? b[1] - a[1]
: ca === 1 ? a[0] - b[0]
: ca === 2 ? a[1] - b[1]
: b[0] - a[0];
}
return function(stream) {
var activeStream = stream,
bufferStream = clipBuffer(),
segments,
polygon,
ring,
x__, y__, v__, // first point
x_, y_, v_, // previous point
first,
clean;
var clipStream = {
point: point,
lineStart: lineStart,
lineEnd: lineEnd,
polygonStart: polygonStart,
polygonEnd: polygonEnd
};
function point(x, y) {
if (visible(x, y)) activeStream.point(x, y);
}
function polygonInside() {
var winding = 0;
for (var i = 0, n = polygon.length; i < n; ++i) {
for (var ring = polygon[i], j = 1, m = ring.length, point = ring[0], a0, a1, b0 = point[0], b1 = point[1]; j < m; ++j) {
a0 = b0, a1 = b1, point = ring[j], b0 = point[0], b1 = point[1];
if (a1 <= y1) { if (b1 > y1 && (b0 - a0) * (y1 - a1) > (b1 - a1) * (x0 - a0)) ++winding; }
else { if (b1 <= y1 && (b0 - a0) * (y1 - a1) < (b1 - a1) * (x0 - a0)) --winding; }
}
}
return winding;
}
// Buffer geometry within a polygon and then clip it en masse.
function polygonStart() {
activeStream = bufferStream, segments = [], polygon = [], clean = true;
}
function polygonEnd() {
var startInside = polygonInside(),
cleanInside = clean && startInside,
visible = (segments = merge(segments)).length;
if (cleanInside || visible) {
stream.polygonStart();
if (cleanInside) {
stream.lineStart();
interpolate(null, null, 1, stream);
stream.lineEnd();
}
if (visible) {
clipPolygon(segments, compareIntersection, startInside, interpolate, stream);
}
stream.polygonEnd();
}
activeStream = stream, segments = polygon = ring = null;
}
function lineStart() {
clipStream.point = linePoint;
if (polygon) polygon.push(ring = []);
first = true;
v_ = false;
x_ = y_ = NaN;
}
// TODO rather than special-case polygons, simply handle them separately.
// Ideally, coincident intersection points should be jittered to avoid
// clipping issues.
function lineEnd() {
if (segments) {
linePoint(x__, y__);
if (v__ && v_) bufferStream.rejoin();
segments.push(bufferStream.result());
}
clipStream.point = point;
if (v_) activeStream.lineEnd();
}
function linePoint(x, y) {
var v = visible(x, y);
if (polygon) ring.push([x, y]);
if (first) {
x__ = x, y__ = y, v__ = v;
first = false;
if (v) {
activeStream.lineStart();
activeStream.point(x, y);
}
} else {
if (v && v_) activeStream.point(x, y);
else {
var a = [x_ = Math.max(clipMin, Math.min(clipMax, x_)), y_ = Math.max(clipMin, Math.min(clipMax, y_))],
b = [x = Math.max(clipMin, Math.min(clipMax, x)), y = Math.max(clipMin, Math.min(clipMax, y))];
if (clipLine(a, b, x0, y0, x1, y1)) {
if (!v_) {
activeStream.lineStart();
activeStream.point(a[0], a[1]);
}
activeStream.point(b[0], b[1]);
if (!v) activeStream.lineEnd();
clean = false;
} else if (v) {
activeStream.lineStart();
activeStream.point(x, y);
clean = false;
}
}
}
x_ = x, y_ = y, v_ = v;
}
return clipStream;
};
}
function extent$1() {
var x0 = 0,
y0 = 0,
x1 = 960,
y1 = 500,
cache,
cacheStream,
clip;
return clip = {
stream: function(stream) {
return cache && cacheStream === stream ? cache : cache = clipExtent(x0, y0, x1, y1)(cacheStream = stream);
},
extent: function(_) {
return arguments.length ? (x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1], cache = cacheStream = null, clip) : [[x0, y0], [x1, y1]];
}
};
}
var lengthSum = adder();
var lambda0$2;
var sinPhi0$1;
var cosPhi0$1;
var lengthStream = {
sphere: noop$2,
point: noop$2,
lineStart: lengthLineStart,
lineEnd: noop$2,
polygonStart: noop$2,
polygonEnd: noop$2
};
function lengthLineStart() {
lengthStream.point = lengthPointFirst;
lengthStream.lineEnd = lengthLineEnd;
}
function lengthLineEnd() {
lengthStream.point = lengthStream.lineEnd = noop$2;
}
function lengthPointFirst(lambda, phi) {
lambda *= radians, phi *= radians;
lambda0$2 = lambda, sinPhi0$1 = sin(phi), cosPhi0$1 = cos(phi);
lengthStream.point = lengthPoint;
}
function lengthPoint(lambda, phi) {
lambda *= radians, phi *= radians;
var sinPhi = sin(phi),
cosPhi = cos(phi),
delta = abs(lambda - lambda0$2),
cosDelta = cos(delta),
sinDelta = sin(delta),
x = cosPhi * sinDelta,
y = cosPhi0$1 * sinPhi - sinPhi0$1 * cosPhi * cosDelta,
z = sinPhi0$1 * sinPhi + cosPhi0$1 * cosPhi * cosDelta;
lengthSum.add(atan2(sqrt$1(x * x + y * y), z));
lambda0$2 = lambda, sinPhi0$1 = sinPhi, cosPhi0$1 = cosPhi;
}
function length$2(object) {
lengthSum.reset();
geoStream(object, lengthStream);
return +lengthSum;
}
var coordinates = [null, null];
var object$1 = {type: "LineString", coordinates: coordinates};
function distance(a, b) {
coordinates[0] = a;
coordinates[1] = b;
return length$2(object$1);
}
function graticuleX(y0, y1, dy) {
var y = range(y0, y1 - epsilon$4, dy).concat(y1);
return function(x) { return y.map(function(y) { return [x, y]; }); };
}
function graticuleY(x0, x1, dx) {
var x = range(x0, x1 - epsilon$4, dx).concat(x1);
return function(y) { return x.map(function(x) { return [x, y]; }); };
}
function graticule() {
var x1, x0, X1, X0,
y1, y0, Y1, Y0,
dx = 10, dy = dx, DX = 90, DY = 360,
x, y, X, Y,
precision = 2.5;
function graticule() {
return {type: "MultiLineString", coordinates: lines()};
}
function lines() {
return range(ceil(X0 / DX) * DX, X1, DX).map(X)
.concat(range(ceil(Y0 / DY) * DY, Y1, DY).map(Y))
.concat(range(ceil(x0 / dx) * dx, x1, dx).filter(function(x) { return abs(x % DX) > epsilon$4; }).map(x))
.concat(range(ceil(y0 / dy) * dy, y1, dy).filter(function(y) { return abs(y % DY) > epsilon$4; }).map(y));
}
graticule.lines = function() {
return lines().map(function(coordinates) { return {type: "LineString", coordinates: coordinates}; });
};
graticule.outline = function() {
return {
type: "Polygon",
coordinates: [
X(X0).concat(
Y(Y1).slice(1),
X(X1).reverse().slice(1),
Y(Y0).reverse().slice(1))
]
};
};
graticule.extent = function(_) {
if (!arguments.length) return graticule.extentMinor();
return graticule.extentMajor(_).extentMinor(_);
};
graticule.extentMajor = function(_) {
if (!arguments.length) return [[X0, Y0], [X1, Y1]];
X0 = +_[0][0], X1 = +_[1][0];
Y0 = +_[0][1], Y1 = +_[1][1];
if (X0 > X1) _ = X0, X0 = X1, X1 = _;
if (Y0 > Y1) _ = Y0, Y0 = Y1, Y1 = _;
return graticule.precision(precision);
};
graticule.extentMinor = function(_) {
if (!arguments.length) return [[x0, y0], [x1, y1]];
x0 = +_[0][0], x1 = +_[1][0];
y0 = +_[0][1], y1 = +_[1][1];
if (x0 > x1) _ = x0, x0 = x1, x1 = _;
if (y0 > y1) _ = y0, y0 = y1, y1 = _;
return graticule.precision(precision);
};
graticule.step = function(_) {
if (!arguments.length) return graticule.stepMinor();
return graticule.stepMajor(_).stepMinor(_);
};
graticule.stepMajor = function(_) {
if (!arguments.length) return [DX, DY];
DX = +_[0], DY = +_[1];
return graticule;
};
graticule.stepMinor = function(_) {
if (!arguments.length) return [dx, dy];
dx = +_[0], dy = +_[1];
return graticule;
};
graticule.precision = function(_) {
if (!arguments.length) return precision;
precision = +_;
x = graticuleX(y0, y1, 90);
y = graticuleY(x0, x1, precision);
X = graticuleX(Y0, Y1, 90);
Y = graticuleY(X0, X1, precision);
return graticule;
};
return graticule
.extentMajor([[-180, -90 + epsilon$4], [180, 90 - epsilon$4]])
.extentMinor([[-180, -80 - epsilon$4], [180, 80 + epsilon$4]]);
}
function interpolate$2(a, b) {
var x0 = a[0] * radians,
y0 = a[1] * radians,
x1 = b[0] * radians,
y1 = b[1] * radians,
cy0 = cos(y0),
sy0 = sin(y0),
cy1 = cos(y1),
sy1 = sin(y1),
kx0 = cy0 * cos(x0),
ky0 = cy0 * sin(x0),
kx1 = cy1 * cos(x1),
ky1 = cy1 * sin(x1),
d = 2 * asin$1(sqrt$1(haversin(y1 - y0) + cy0 * cy1 * haversin(x1 - x0))),
k = sin(d);
var interpolate = d ? function(t) {
var B = sin(t *= d) / k,
A = sin(d - t) / k,
x = A * kx0 + B * kx1,
y = A * ky0 + B * ky1,
z = A * sy0 + B * sy1;
return [
atan2(y, x) * degrees$1,
atan2(z, sqrt$1(x * x + y * y)) * degrees$1
];
} : function() {
return [x0 * degrees$1, y0 * degrees$1];
};
interpolate.distance = d;
return interpolate;
}
function identity$7(x) {
return x;
}
var areaSum$1 = adder();
var areaRingSum$1 = adder();
var x00;
var y00;
var x0$1;
var y0$1;
var areaStream$1 = {
point: noop$2,
lineStart: noop$2,
lineEnd: noop$2,
polygonStart: function() {
areaStream$1.lineStart = areaRingStart$1;
areaStream$1.lineEnd = areaRingEnd$1;
},
polygonEnd: function() {
areaStream$1.lineStart = areaStream$1.lineEnd = areaStream$1.point = noop$2;
areaSum$1.add(abs(areaRingSum$1));
areaRingSum$1.reset();
},
result: function() {
var area = areaSum$1 / 2;
areaSum$1.reset();
return area;
}
};
function areaRingStart$1() {
areaStream$1.point = areaPointFirst$1;
}
function areaPointFirst$1(x, y) {
areaStream$1.point = areaPoint$1;
x00 = x0$1 = x, y00 = y0$1 = y;
}
function areaPoint$1(x, y) {
areaRingSum$1.add(y0$1 * x - x0$1 * y);
x0$1 = x, y0$1 = y;
}
function areaRingEnd$1() {
areaPoint$1(x00, y00);
}
var x0$2 = Infinity;
var y0$2 = x0$2;
var x1 = -x0$2;
var y1 = x1;
var boundsStream$1 = {
point: boundsPoint$1,
lineStart: noop$2,
lineEnd: noop$2,
polygonStart: noop$2,
polygonEnd: noop$2,
result: function() {
var bounds = [[x0$2, y0$2], [x1, y1]];
x1 = y1 = -(y0$2 = x0$2 = Infinity);
return bounds;
}
};
function boundsPoint$1(x, y) {
if (x < x0$2) x0$2 = x;
if (x > x1) x1 = x;
if (y < y0$2) y0$2 = y;
if (y > y1) y1 = y;
}
var X0$1 = 0;
var Y0$1 = 0;
var Z0$1 = 0;
var X1$1 = 0;
var Y1$1 = 0;
var Z1$1 = 0;
var X2$1 = 0;
var Y2$1 = 0;
var Z2$1 = 0;
var x00$1;
var y00$1;
var x0$3;
var y0$3;
var centroidStream$1 = {
point: centroidPoint$1,
lineStart: centroidLineStart$1,
lineEnd: centroidLineEnd$1,
polygonStart: function() {
centroidStream$1.lineStart = centroidRingStart$1;
centroidStream$1.lineEnd = centroidRingEnd$1;
},
polygonEnd: function() {
centroidStream$1.point = centroidPoint$1;
centroidStream$1.lineStart = centroidLineStart$1;
centroidStream$1.lineEnd = centroidLineEnd$1;
},
result: function() {
var centroid = Z2$1 ? [X2$1 / Z2$1, Y2$1 / Z2$1]
: Z1$1 ? [X1$1 / Z1$1, Y1$1 / Z1$1]
: Z0$1 ? [X0$1 / Z0$1, Y0$1 / Z0$1]
: [NaN, NaN];
X0$1 = Y0$1 = Z0$1 =
X1$1 = Y1$1 = Z1$1 =
X2$1 = Y2$1 = Z2$1 = 0;
return centroid;
}
};
function centroidPoint$1(x, y) {
X0$1 += x;
Y0$1 += y;
++Z0$1;
}
function centroidLineStart$1() {
centroidStream$1.point = centroidPointFirstLine;
}
function centroidPointFirstLine(x, y) {
centroidStream$1.point = centroidPointLine;
centroidPoint$1(x0$3 = x, y0$3 = y);
}
function centroidPointLine(x, y) {
var dx = x - x0$3, dy = y - y0$3, z = sqrt$1(dx * dx + dy * dy);
X1$1 += z * (x0$3 + x) / 2;
Y1$1 += z * (y0$3 + y) / 2;
Z1$1 += z;
centroidPoint$1(x0$3 = x, y0$3 = y);
}
function centroidLineEnd$1() {
centroidStream$1.point = centroidPoint$1;
}
function centroidRingStart$1() {
centroidStream$1.point = centroidPointFirstRing;
}
function centroidRingEnd$1() {
centroidPointRing(x00$1, y00$1);
}
function centroidPointFirstRing(x, y) {
centroidStream$1.point = centroidPointRing;
centroidPoint$1(x00$1 = x0$3 = x, y00$1 = y0$3 = y);
}
function centroidPointRing(x, y) {
var dx = x - x0$3,
dy = y - y0$3,
z = sqrt$1(dx * dx + dy * dy);
X1$1 += z * (x0$3 + x) / 2;
Y1$1 += z * (y0$3 + y) / 2;
Z1$1 += z;
z = y0$3 * x - x0$3 * y;
X2$1 += z * (x0$3 + x);
Y2$1 += z * (y0$3 + y);
Z2$1 += z * 3;
centroidPoint$1(x0$3 = x, y0$3 = y);
}
function PathContext(context) {
var pointRadius = 4.5;
var stream = {
point: point,
// While inside a line, override point to moveTo then lineTo.
lineStart: function() { stream.point = pointLineStart; },
lineEnd: lineEnd,
// While inside a polygon, override lineEnd to closePath.
polygonStart: function() { stream.lineEnd = lineEndPolygon; },
polygonEnd: function() { stream.lineEnd = lineEnd; stream.point = point; },
pointRadius: function(_) {
pointRadius = _;
return stream;
},
result: noop$2
};
function point(x, y) {
context.moveTo(x + pointRadius, y);
context.arc(x, y, pointRadius, 0, tau$3);
}
function pointLineStart(x, y) {
context.moveTo(x, y);
stream.point = pointLine;
}
function pointLine(x, y) {
context.lineTo(x, y);
}
function lineEnd() {
stream.point = point;
}
function lineEndPolygon() {
context.closePath();
}
return stream;
}
function PathString() {
var pointCircle = circle$2(4.5),
string = [];
var stream = {
point: point,
lineStart: lineStart,
lineEnd: lineEnd,
polygonStart: function() {
stream.lineEnd = lineEndPolygon;
},
polygonEnd: function() {
stream.lineEnd = lineEnd;
stream.point = point;
},
pointRadius: function(_) {
pointCircle = circle$2(_);
return stream;
},
result: function() {
if (string.length) {
var result = string.join("");
string = [];
return result;
}
}
};
function point(x, y) {
string.push("M", x, ",", y, pointCircle);
}
function pointLineStart(x, y) {
string.push("M", x, ",", y);
stream.point = pointLine;
}
function pointLine(x, y) {
string.push("L", x, ",", y);
}
function lineStart() {
stream.point = pointLineStart;
}
function lineEnd() {
stream.point = point;
}
function lineEndPolygon() {
string.push("Z");
}
return stream;
}
function circle$2(radius) {
return "m0," + radius
+ "a" + radius + "," + radius + " 0 1,1 0," + -2 * radius
+ "a" + radius + "," + radius + " 0 1,1 0," + 2 * radius
+ "z";
}
function index$3() {
var pointRadius = 4.5,
projection,
projectionStream,
context,
contextStream;
function path(object) {
if (object) {
if (typeof pointRadius === "function") contextStream.pointRadius(+pointRadius.apply(this, arguments));
geoStream(object, projectionStream(contextStream));
}
return contextStream.result();
}
path.area = function(object) {
geoStream(object, projectionStream(areaStream$1));
return areaStream$1.result();
};
path.bounds = function(object) {
geoStream(object, projectionStream(boundsStream$1));
return boundsStream$1.result();
};
path.centroid = function(object) {
geoStream(object, projectionStream(centroidStream$1));
return centroidStream$1.result();
};
path.projection = function(_) {
return arguments.length ? (projectionStream = (projection = _) == null ? identity$7 : _.stream, path) : projection;
};
path.context = function(_) {
if (!arguments.length) return context;
contextStream = (context = _) == null ? new PathString : new PathContext(_);
if (typeof pointRadius !== "function") contextStream.pointRadius(pointRadius);
return path;
};
path.pointRadius = function(_) {
if (!arguments.length) return pointRadius;
pointRadius = typeof _ === "function" ? _ : (contextStream.pointRadius(+_), +_);
return path;
};
return path.projection(null).context(null);
}
var sum$2 = adder();
function polygonContains(polygon, point) {
var lambda = point[0],
phi = point[1],
normal = [sin(lambda), -cos(lambda), 0],
angle = 0,
winding = 0;
sum$2.reset();
for (var i = 0, n = polygon.length; i < n; ++i) {
if (!(m = (ring = polygon[i]).length)) continue;
var ring,
m,
point0 = ring[m - 1],
lambda0 = point0[0],
phi0 = point0[1] / 2 + quarterPi,
sinPhi0 = sin(phi0),
cosPhi0 = cos(phi0);
for (var j = 0; j < m; ++j, lambda0 = lambda1, sinPhi0 = sinPhi1, cosPhi0 = cosPhi1, point0 = point1) {
var point1 = ring[j],
lambda1 = point1[0],
phi1 = point1[1] / 2 + quarterPi,
sinPhi1 = sin(phi1),
cosPhi1 = cos(phi1),
delta = lambda1 - lambda0,
sign = delta >= 0 ? 1 : -1,
absDelta = sign * delta,
antimeridian = absDelta > pi$3,
k = sinPhi0 * sinPhi1;
sum$2.add(atan2(k * sign * sin(absDelta), cosPhi0 * cosPhi1 + k * cos(absDelta)));
angle += antimeridian ? delta + sign * tau$3 : delta;
// Are the longitudes either side of the point’s meridian (lambda),
// and are the latitudes smaller than the parallel (phi)?
if (antimeridian ^ lambda0 >= lambda ^ lambda1 >= lambda) {
var arc = cartesianCross(cartesian(point0), cartesian(point1));
cartesianNormalizeInPlace(arc);
var intersection = cartesianCross(normal, arc);
cartesianNormalizeInPlace(intersection);
var phiArc = (antimeridian ^ delta >= 0 ? -1 : 1) * asin$1(intersection[2]);
if (phi > phiArc || phi === phiArc && (arc[0] || arc[1])) {
winding += antimeridian ^ delta >= 0 ? 1 : -1;
}
}
}
}
// First, determine whether the South pole is inside or outside:
//
// It is inside if:
// * the polygon winds around it in a clockwise direction.
// * the polygon does not (cumulatively) wind around it, but has a negative
// (counter-clockwise) area.
//
// Second, count the (signed) number of times a segment crosses a lambda
// from the point to the South pole. If it is zero, then the point is the
// same side as the South pole.
return (angle < -epsilon$4 || angle < epsilon$4 && sum$2 < -epsilon$4) ^ (winding & 1);
}
function clip(pointVisible, clipLine, interpolate, start) {
return function(rotate, sink) {
var line = clipLine(sink),
rotatedStart = rotate.invert(start[0], start[1]),
ringBuffer = clipBuffer(),
ringSink = clipLine(ringBuffer),
polygonStarted = false,
polygon,
segments,
ring;
var clip = {
point: point,
lineStart: lineStart,
lineEnd: lineEnd,
polygonStart: function() {
clip.point = pointRing;
clip.lineStart = ringStart;
clip.lineEnd = ringEnd;
segments = [];
polygon = [];
},
polygonEnd: function() {
clip.point = point;
clip.lineStart = lineStart;
clip.lineEnd = lineEnd;
segments = merge(segments);
var startInside = polygonContains(polygon, rotatedStart);
if (segments.length) {
if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
clipPolygon(segments, compareIntersection, startInside, interpolate, sink);
} else if (startInside) {
if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
sink.lineStart();
interpolate(null, null, 1, sink);
sink.lineEnd();
}
if (polygonStarted) sink.polygonEnd(), polygonStarted = false;
segments = polygon = null;
},
sphere: function() {
sink.polygonStart();
sink.lineStart();
interpolate(null, null, 1, sink);
sink.lineEnd();
sink.polygonEnd();
}
};
function point(lambda, phi) {
var point = rotate(lambda, phi);
if (pointVisible(lambda = point[0], phi = point[1])) sink.point(lambda, phi);
}
function pointLine(lambda, phi) {
var point = rotate(lambda, phi);
line.point(point[0], point[1]);
}
function lineStart() {
clip.point = pointLine;
line.lineStart();
}
function lineEnd() {
clip.point = point;
line.lineEnd();
}
function pointRing(lambda, phi) {
ring.push([lambda, phi]);
var point = rotate(lambda, phi);
ringSink.point(point[0], point[1]);
}
function ringStart() {
ringSink.lineStart();
ring = [];
}
function ringEnd() {
pointRing(ring[0][0], ring[0][1]);
ringSink.lineEnd();
var clean = ringSink.clean(),
ringSegments = ringBuffer.result(),
i, n = ringSegments.length, m,
segment,
point;
ring.pop();
polygon.push(ring);
ring = null;
if (!n) return;
// No intersections.
if (clean & 1) {
segment = ringSegments[0];
if ((m = segment.length - 1) > 0) {
if (!polygonStarted) sink.polygonStart(), polygonStarted = true;
sink.lineStart();
for (i = 0; i < m; ++i) sink.point((point = segment[i])[0], point[1]);
sink.lineEnd();
}
return;
}
// Rejoin connected segments.
// TODO reuse ringBuffer.rejoin()?
if (n > 1 && clean & 2) ringSegments.push(ringSegments.pop().concat(ringSegments.shift()));
segments.push(ringSegments.filter(validSegment));
}
return clip;
};
}
function validSegment(segment) {
return segment.length > 1;
}
// Intersections are sorted along the clip edge. For both antimeridian cutting
// and circle clipping, the same comparison is used.
function compareIntersection(a, b) {
return ((a = a.x)[0] < 0 ? a[1] - halfPi$2 - epsilon$4 : halfPi$2 - a[1])
- ((b = b.x)[0] < 0 ? b[1] - halfPi$2 - epsilon$4 : halfPi$2 - b[1]);
}
var clipAntimeridian = clip(
function() { return true; },
clipAntimeridianLine,
clipAntimeridianInterpolate,
[-pi$3, -halfPi$2]
);
// Takes a line and cuts into visible segments. Return values: 0 - there were
// intersections or the line was empty; 1 - no intersections; 2 - there were
// intersections, and the first and last segments should be rejoined.
function clipAntimeridianLine(stream) {
var lambda0 = NaN,
phi0 = NaN,
sign0 = NaN,
clean; // no intersections
return {
lineStart: function() {
stream.lineStart();
clean = 1;
},
point: function(lambda1, phi1) {
var sign1 = lambda1 > 0 ? pi$3 : -pi$3,
delta = abs(lambda1 - lambda0);
if (abs(delta - pi$3) < epsilon$4) { // line crosses a pole
stream.point(lambda0, phi0 = (phi0 + phi1) / 2 > 0 ? halfPi$2 : -halfPi$2);
stream.point(sign0, phi0);
stream.lineEnd();
stream.lineStart();
stream.point(sign1, phi0);
stream.point(lambda1, phi0);
clean = 0;
} else if (sign0 !== sign1 && delta >= pi$3) { // line crosses antimeridian
if (abs(lambda0 - sign0) < epsilon$4) lambda0 -= sign0 * epsilon$4; // handle degeneracies
if (abs(lambda1 - sign1) < epsilon$4) lambda1 -= sign1 * epsilon$4;
phi0 = clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1);
stream.point(sign0, phi0);
stream.lineEnd();
stream.lineStart();
stream.point(sign1, phi0);
clean = 0;
}
stream.point(lambda0 = lambda1, phi0 = phi1);
sign0 = sign1;
},
lineEnd: function() {
stream.lineEnd();
lambda0 = phi0 = NaN;
},
clean: function() {
return 2 - clean; // if intersections, rejoin first and last segments
}
};
}
function clipAntimeridianIntersect(lambda0, phi0, lambda1, phi1) {
var cosPhi0,
cosPhi1,
sinLambda0Lambda1 = sin(lambda0 - lambda1);
return abs(sinLambda0Lambda1) > epsilon$4
? atan((sin(phi0) * (cosPhi1 = cos(phi1)) * sin(lambda1)
- sin(phi1) * (cosPhi0 = cos(phi0)) * sin(lambda0))
/ (cosPhi0 * cosPhi1 * sinLambda0Lambda1))
: (phi0 + phi1) / 2;
}
function clipAntimeridianInterpolate(from, to, direction, stream) {
var phi;
if (from == null) {
phi = direction * halfPi$2;
stream.point(-pi$3, phi);
stream.point(0, phi);
stream.point(pi$3, phi);
stream.point(pi$3, 0);
stream.point(pi$3, -phi);
stream.point(0, -phi);
stream.point(-pi$3, -phi);
stream.point(-pi$3, 0);
stream.point(-pi$3, phi);
} else if (abs(from[0] - to[0]) > epsilon$4) {
var lambda = from[0] < to[0] ? pi$3 : -pi$3;
phi = direction * lambda / 2;
stream.point(-lambda, phi);
stream.point(0, phi);
stream.point(lambda, phi);
} else {
stream.point(to[0], to[1]);
}
}
function clipCircle(radius, delta) {
var cr = cos(radius),
smallRadius = cr > 0,
notHemisphere = abs(cr) > epsilon$4; // TODO optimise for this common case
function interpolate(from, to, direction, stream) {
circleStream(stream, radius, delta, direction, from, to);
}
function visible(lambda, phi) {
return cos(lambda) * cos(phi) > cr;
}
// Takes a line and cuts into visible segments. Return values used for polygon
// clipping: 0 - there were intersections or the line was empty; 1 - no
// intersections 2 - there were intersections, and the first and last segments
// should be rejoined.
function clipLine(stream) {
var point0, // previous point
c0, // code for previous point
v0, // visibility of previous point
v00, // visibility of first point
clean; // no intersections
return {
lineStart: function() {
v00 = v0 = false;
clean = 1;
},
point: function(lambda, phi) {
var point1 = [lambda, phi],
point2,
v = visible(lambda, phi),
c = smallRadius
? v ? 0 : code(lambda, phi)
: v ? code(lambda + (lambda < 0 ? pi$3 : -pi$3), phi) : 0;
if (!point0 && (v00 = v0 = v)) stream.lineStart();
// Handle degeneracies.
// TODO ignore if not clipping polygons.
if (v !== v0) {
point2 = intersect(point0, point1);
if (pointEqual(point0, point2) || pointEqual(point1, point2)) {
point1[0] += epsilon$4;
point1[1] += epsilon$4;
v = visible(point1[0], point1[1]);
}
}
if (v !== v0) {
clean = 0;
if (v) {
// outside going in
stream.lineStart();
point2 = intersect(point1, point0);
stream.point(point2[0], point2[1]);
} else {
// inside going out
point2 = intersect(point0, point1);
stream.point(point2[0], point2[1]);
stream.lineEnd();
}
point0 = point2;
} else if (notHemisphere && point0 && smallRadius ^ v) {
var t;
// If the codes for two points are different, or are both zero,
// and there this segment intersects with the small circle.
if (!(c & c0) && (t = intersect(point1, point0, true))) {
clean = 0;
if (smallRadius) {
stream.lineStart();
stream.point(t[0][0], t[0][1]);
stream.point(t[1][0], t[1][1]);
stream.lineEnd();
} else {
stream.point(t[1][0], t[1][1]);
stream.lineEnd();
stream.lineStart();
stream.point(t[0][0], t[0][1]);
}
}
}
if (v && (!point0 || !pointEqual(point0, point1))) {
stream.point(point1[0], point1[1]);
}
point0 = point1, v0 = v, c0 = c;
},
lineEnd: function() {
if (v0) stream.lineEnd();
point0 = null;
},
// Rejoin first and last segments if there were intersections and the first
// and last points were visible.
clean: function() {
return clean | ((v00 && v0) << 1);
}
};
}
// Intersects the great circle between a and b with the clip circle.
function intersect(a, b, two) {
var pa = cartesian(a),
pb = cartesian(b);
// We have two planes, n1.p = d1 and n2.p = d2.
// Find intersection line p(t) = c1 n1 + c2 n2 + t (n1 ⨯ n2).
var n1 = [1, 0, 0], // normal
n2 = cartesianCross(pa, pb),
n2n2 = cartesianDot(n2, n2),
n1n2 = n2[0], // cartesianDot(n1, n2),
determinant = n2n2 - n1n2 * n1n2;
// Two polar points.
if (!determinant) return !two && a;
var c1 = cr * n2n2 / determinant,
c2 = -cr * n1n2 / determinant,
n1xn2 = cartesianCross(n1, n2),
A = cartesianScale(n1, c1),
B = cartesianScale(n2, c2);
cartesianAddInPlace(A, B);
// Solve |p(t)|^2 = 1.
var u = n1xn2,
w = cartesianDot(A, u),
uu = cartesianDot(u, u),
t2 = w * w - uu * (cartesianDot(A, A) - 1);
if (t2 < 0) return;
var t = sqrt$1(t2),
q = cartesianScale(u, (-w - t) / uu);
cartesianAddInPlace(q, A);
q = spherical(q);
if (!two) return q;
// Two intersection points.
var lambda0 = a[0],
lambda1 = b[0],
phi0 = a[1],
phi1 = b[1],
z;
if (lambda1 < lambda0) z = lambda0, lambda0 = lambda1, lambda1 = z;
var delta = lambda1 - lambda0,
polar = abs(delta - pi$3) < epsilon$4,
meridian = polar || delta < epsilon$4;
if (!polar && phi1 < phi0) z = phi0, phi0 = phi1, phi1 = z;
// Check that the first point is between a and b.
if (meridian
? polar
? phi0 + phi1 > 0 ^ q[1] < (abs(q[0] - lambda0) < epsilon$4 ? phi0 : phi1)
: phi0 <= q[1] && q[1] <= phi1
: delta > pi$3 ^ (lambda0 <= q[0] && q[0] <= lambda1)) {
var q1 = cartesianScale(u, (-w + t) / uu);
cartesianAddInPlace(q1, A);
return [q, spherical(q1)];
}
}
// Generates a 4-bit vector representing the location of a point relative to
// the small circle's bounding box.
function code(lambda, phi) {
var r = smallRadius ? radius : pi$3 - radius,
code = 0;
if (lambda < -r) code |= 1; // left
else if (lambda > r) code |= 2; // right
if (phi < -r) code |= 4; // below
else if (phi > r) code |= 8; // above
return code;
}
return clip(visible, clipLine, interpolate, smallRadius ? [0, -radius] : [-pi$3, radius - pi$3]);
}
function transform$1(prototype) {
return {
stream: transform$2(prototype)
};
}
function transform$2(prototype) {
function T() {}
var p = T.prototype = Object.create(Transform$1.prototype);
for (var k in prototype) p[k] = prototype[k];
return function(stream) {
var t = new T;
t.stream = stream;
return t;
};
}
function Transform$1() {}
Transform$1.prototype = {
point: function(x, y) { this.stream.point(x, y); },
sphere: function() { this.stream.sphere(); },
lineStart: function() { this.stream.lineStart(); },
lineEnd: function() { this.stream.lineEnd(); },
polygonStart: function() { this.stream.polygonStart(); },
polygonEnd: function() { this.stream.polygonEnd(); }
};
function fit(project, extent, object) {
var w = extent[1][0] - extent[0][0],
h = extent[1][1] - extent[0][1],
clip = project.clipExtent && project.clipExtent();
project
.scale(150)
.translate([0, 0]);
if (clip != null) project.clipExtent(null);
geoStream(object, project.stream(boundsStream$1));
var b = boundsStream$1.result(),
k = Math.min(w / (b[1][0] - b[0][0]), h / (b[1][1] - b[0][1])),
x = +extent[0][0] + (w - k * (b[1][0] + b[0][0])) / 2,
y = +extent[0][1] + (h - k * (b[1][1] + b[0][1])) / 2;
if (clip != null) project.clipExtent(clip);
return project
.scale(k * 150)
.translate([x, y]);
}
function fitSize(project) {
return function(size, object) {
return fit(project, [[0, 0], size], object);
};
}
function fitExtent(project) {
return function(extent, object) {
return fit(project, extent, object);
};
}
var maxDepth = 16;
var cosMinDistance = cos(30 * radians);
// cos(minimum angular distance)
function resample(project, delta2) {
return +delta2 ? resample$1(project, delta2) : resampleNone(project);
}
function resampleNone(project) {
return transform$2({
point: function(x, y) {
x = project(x, y);
this.stream.point(x[0], x[1]);
}
});
}
function resample$1(project, delta2) {
function resampleLineTo(x0, y0, lambda0, a0, b0, c0, x1, y1, lambda1, a1, b1, c1, depth, stream) {
var dx = x1 - x0,
dy = y1 - y0,
d2 = dx * dx + dy * dy;
if (d2 > 4 * delta2 && depth--) {
var a = a0 + a1,
b = b0 + b1,
c = c0 + c1,
m = sqrt$1(a * a + b * b + c * c),
phi2 = asin$1(c /= m),
lambda2 = abs(abs(c) - 1) < epsilon$4 || abs(lambda0 - lambda1) < epsilon$4 ? (lambda0 + lambda1) / 2 : atan2(b, a),
p = project(lambda2, phi2),
x2 = p[0],
y2 = p[1],
dx2 = x2 - x0,
dy2 = y2 - y0,
dz = dy * dx2 - dx * dy2;
if (dz * dz / d2 > delta2 // perpendicular projected distance
|| abs((dx * dx2 + dy * dy2) / d2 - 0.5) > 0.3 // midpoint close to an end
|| a0 * a1 + b0 * b1 + c0 * c1 < cosMinDistance) { // angular distance
resampleLineTo(x0, y0, lambda0, a0, b0, c0, x2, y2, lambda2, a /= m, b /= m, c, depth, stream);
stream.point(x2, y2);
resampleLineTo(x2, y2, lambda2, a, b, c, x1, y1, lambda1, a1, b1, c1, depth, stream);
}
}
}
return function(stream) {
var lambda00, x00, y00, a00, b00, c00, // first point
lambda0, x0, y0, a0, b0, c0; // previous point
var resampleStream = {
point: point,
lineStart: lineStart,
lineEnd: lineEnd,
polygonStart: function() { stream.polygonStart(); resampleStream.lineStart = ringStart; },
polygonEnd: function() { stream.polygonEnd(); resampleStream.lineStart = lineStart; }
};
function point(x, y) {
x = project(x, y);
stream.point(x[0], x[1]);
}
function lineStart() {
x0 = NaN;
resampleStream.point = linePoint;
stream.lineStart();
}
function linePoint(lambda, phi) {
var c = cartesian([lambda, phi]), p = project(lambda, phi);
resampleLineTo(x0, y0, lambda0, a0, b0, c0, x0 = p[0], y0 = p[1], lambda0 = lambda, a0 = c[0], b0 = c[1], c0 = c[2], maxDepth, stream);
stream.point(x0, y0);
}
function lineEnd() {
resampleStream.point = point;
stream.lineEnd();
}
function ringStart() {
lineStart();
resampleStream.point = ringPoint;
resampleStream.lineEnd = ringEnd;
}
function ringPoint(lambda, phi) {
linePoint(lambda00 = lambda, phi), x00 = x0, y00 = y0, a00 = a0, b00 = b0, c00 = c0;
resampleStream.point = linePoint;
}
function ringEnd() {
resampleLineTo(x0, y0, lambda0, a0, b0, c0, x00, y00, lambda00, a00, b00, c00, maxDepth, stream);
resampleStream.lineEnd = lineEnd;
lineEnd();
}
return resampleStream;
};
}
var transformRadians = transform$2({
point: function(x, y) {
this.stream.point(x * radians, y * radians);
}
});
function projection(project) {
return projectionMutator(function() { return project; })();
}
function projectionMutator(projectAt) {
var project,
k = 150, // scale
x = 480, y = 250, // translate
dx, dy, lambda = 0, phi = 0, // center
deltaLambda = 0, deltaPhi = 0, deltaGamma = 0, rotate, projectRotate, // rotate
theta = null, preclip = clipAntimeridian, // clip angle
x0 = null, y0, x1, y1, postclip = identity$7, // clip extent
delta2 = 0.5, projectResample = resample(projectTransform, delta2), // precision
cache,
cacheStream;
function projection(point) {
point = projectRotate(point[0] * radians, point[1] * radians);
return [point[0] * k + dx, dy - point[1] * k];
}
function invert(point) {
point = projectRotate.invert((point[0] - dx) / k, (dy - point[1]) / k);
return point && [point[0] * degrees$1, point[1] * degrees$1];
}
function projectTransform(x, y) {
return x = project(x, y), [x[0] * k + dx, dy - x[1] * k];
}
projection.stream = function(stream) {
return cache && cacheStream === stream ? cache : cache = transformRadians(preclip(rotate, projectResample(postclip(cacheStream = stream))));
};
projection.clipAngle = function(_) {
return arguments.length ? (preclip = +_ ? clipCircle(theta = _ * radians, 6 * radians) : (theta = null, clipAntimeridian), reset()) : theta * degrees$1;
};
projection.clipExtent = function(_) {
return arguments.length ? (postclip = _ == null ? (x0 = y0 = x1 = y1 = null, identity$7) : clipExtent(x0 = +_[0][0], y0 = +_[0][1], x1 = +_[1][0], y1 = +_[1][1]), reset()) : x0 == null ? null : [[x0, y0], [x1, y1]];
};
projection.scale = function(_) {
return arguments.length ? (k = +_, recenter()) : k;
};
projection.translate = function(_) {
return arguments.length ? (x = +_[0], y = +_[1], recenter()) : [x, y];
};
projection.center = function(_) {
return arguments.length ? (lambda = _[0] % 360 * radians, phi = _[1] % 360 * radians, recenter()) : [lambda * degrees$1, phi * degrees$1];
};
projection.rotate = function(_) {
return arguments.length ? (deltaLambda = _[0] % 360 * radians, deltaPhi = _[1] % 360 * radians, deltaGamma = _.length > 2 ? _[2] % 360 * radians : 0, recenter()) : [deltaLambda * degrees$1, deltaPhi * degrees$1, deltaGamma * degrees$1];
};
projection.precision = function(_) {
return arguments.length ? (projectResample = resample(projectTransform, delta2 = _ * _), reset()) : sqrt$1(delta2);
};
projection.fitExtent = fitExtent(projection);
projection.fitSize = fitSize(projection);
function recenter() {
projectRotate = compose(rotate = rotateRadians(deltaLambda, deltaPhi, deltaGamma), project);
var center = project(lambda, phi);
dx = x - center[0] * k;
dy = y + center[1] * k;
return reset();
}
function reset() {
cache = cacheStream = null;
return projection;
}
return function() {
project = projectAt.apply(this, arguments);
projection.invert = project.invert && invert;
return recenter();
};
}
function conicProjection(projectAt) {
var phi0 = 0,
phi1 = pi$3 / 3,
m = projectionMutator(projectAt),
p = m(phi0, phi1);
p.parallels = function(_) {
return arguments.length ? m(phi0 = _[0] * radians, phi1 = _[1] * radians) : [phi0 * degrees$1, phi1 * degrees$1];
};
return p;
}
function conicEqualAreaRaw(y0, y1) {
var sy0 = sin(y0),
n = (sy0 + sin(y1)) / 2,
c = 1 + sy0 * (2 * n - sy0),
r0 = sqrt$1(c) / n;
function project(x, y) {
var r = sqrt$1(c - 2 * n * sin(y)) / n;
return [r * sin(x *= n), r0 - r * cos(x)];
}
project.invert = function(x, y) {
var r0y = r0 - y;
return [atan2(x, r0y) / n, asin$1((c - (x * x + r0y * r0y) * n * n) / (2 * n))];
};
return project;
}
function conicEqualArea() {
return conicProjection(conicEqualAreaRaw)
.scale(155.424)
.center([0, 33.6442]);
}
function albers() {
return conicEqualArea()
.parallels([29.5, 45.5])
.scale(1070)
.translate([480, 250])
.rotate([96, 0])
.center([-0.6, 38.7]);
}
// The projections must have mutually exclusive clip regions on the sphere,
// as this will avoid emitting interleaving lines and polygons.
function multiplex(streams) {
var n = streams.length;
return {
point: function(x, y) { var i = -1; while (++i < n) streams[i].point(x, y); },
sphere: function() { var i = -1; while (++i < n) streams[i].sphere(); },
lineStart: function() { var i = -1; while (++i < n) streams[i].lineStart(); },
lineEnd: function() { var i = -1; while (++i < n) streams[i].lineEnd(); },
polygonStart: function() { var i = -1; while (++i < n) streams[i].polygonStart(); },
polygonEnd: function() { var i = -1; while (++i < n) streams[i].polygonEnd(); }
};
}
// A composite projection for the United States, configured by default for
// 960×500. The projection also works quite well at 960×600 if you change the
// scale to 1285 and adjust the translate accordingly. The set of standard
// parallels for each region comes from USGS, which is published here:
// http://egsc.usgs.gov/isb/pubs/MapProjections/projections.html#albers
function albersUsa() {
var cache,
cacheStream,
lower48 = albers(), lower48Point,
alaska = conicEqualArea().rotate([154, 0]).center([-2, 58.5]).parallels([55, 65]), alaskaPoint, // EPSG:3338
hawaii = conicEqualArea().rotate([157, 0]).center([-3, 19.9]).parallels([8, 18]), hawaiiPoint, // ESRI:102007
point, pointStream = {point: function(x, y) { point = [x, y]; }};
function albersUsa(coordinates) {
var x = coordinates[0], y = coordinates[1];
return point = null,
(lower48Point.point(x, y), point)
|| (alaskaPoint.point(x, y), point)
|| (hawaiiPoint.point(x, y), point);
}
albersUsa.invert = function(coordinates) {
var k = lower48.scale(),
t = lower48.translate(),
x = (coordinates[0] - t[0]) / k,
y = (coordinates[1] - t[1]) / k;
return (y >= 0.120 && y < 0.234 && x >= -0.425 && x < -0.214 ? alaska
: y >= 0.166 && y < 0.234 && x >= -0.214 && x < -0.115 ? hawaii
: lower48).invert(coordinates);
};
albersUsa.stream = function(stream) {
return cache && cacheStream === stream ? cache : cache = multiplex([lower48.stream(cacheStream = stream), alaska.stream(stream), hawaii.stream(stream)]);
};
albersUsa.precision = function(_) {
if (!arguments.length) return lower48.precision();
lower48.precision(_), alaska.precision(_), hawaii.precision(_);
return albersUsa;
};
albersUsa.scale = function(_) {
if (!arguments.length) return lower48.scale();
lower48.scale(_), alaska.scale(_ * 0.35), hawaii.scale(_);
return albersUsa.translate(lower48.translate());
};
albersUsa.translate = function(_) {
if (!arguments.length) return lower48.translate();
var k = lower48.scale(), x = +_[0], y = +_[1];
lower48Point = lower48
.translate(_)
.clipExtent([[x - 0.455 * k, y - 0.238 * k], [x + 0.455 * k, y + 0.238 * k]])
.stream(pointStream);
alaskaPoint = alaska
.translate([x - 0.307 * k, y + 0.201 * k])
.clipExtent([[x - 0.425 * k + epsilon$4, y + 0.120 * k + epsilon$4], [x - 0.214 * k - epsilon$4, y + 0.234 * k - epsilon$4]])
.stream(pointStream);
hawaiiPoint = hawaii
.translate([x - 0.205 * k, y + 0.212 * k])
.clipExtent([[x - 0.214 * k + epsilon$4, y + 0.166 * k + epsilon$4], [x - 0.115 * k - epsilon$4, y + 0.234 * k - epsilon$4]])
.stream(pointStream);
return albersUsa;
};
albersUsa.fitExtent = fitExtent(albersUsa);
albersUsa.fitSize = fitSize(albersUsa);
return albersUsa.scale(1070);
}
function azimuthalRaw(scale) {
return function(x, y) {
var cx = cos(x),
cy = cos(y),
k = scale(cx * cy);
return [
k * cy * sin(x),
k * sin(y)
];
}
}
function azimuthalInvert(angle) {
return function(x, y) {
var z = sqrt$1(x * x + y * y),
c = angle(z),
sc = sin(c),
cc = cos(c);
return [
atan2(x * sc, z * cc),
asin$1(z && y * sc / z)
];
}
}
var azimuthalEqualAreaRaw = azimuthalRaw(function(cxcy) {
return sqrt$1(2 / (1 + cxcy));
});
azimuthalEqualAreaRaw.invert = azimuthalInvert(function(z) {
return 2 * asin$1(z / 2);
});
function azimuthalEqualArea() {
return projection(azimuthalEqualAreaRaw)
.scale(124.75)
.clipAngle(180 - 1e-3);
}
var azimuthalEquidistantRaw = azimuthalRaw(function(c) {
return (c = acos(c)) && c / sin(c);
});
azimuthalEquidistantRaw.invert = azimuthalInvert(function(z) {
return z;
});
function azimuthalEquidistant() {
return projection(azimuthalEquidistantRaw)
.scale(79.4188)
.clipAngle(180 - 1e-3);
}
function mercatorRaw(lambda, phi) {
return [lambda, log$1(tan((halfPi$2 + phi) / 2))];
}
mercatorRaw.invert = function(x, y) {
return [x, 2 * atan(exp(y)) - halfPi$2];
};
function mercator() {
return mercatorProjection(mercatorRaw)
.scale(961 / tau$3);
}
function mercatorProjection(project) {
var m = projection(project),
scale = m.scale,
translate = m.translate,
clipExtent = m.clipExtent,
clipAuto;
m.scale = function(_) {
return arguments.length ? (scale(_), clipAuto && m.clipExtent(null), m) : scale();
};
m.translate = function(_) {
return arguments.length ? (translate(_), clipAuto && m.clipExtent(null), m) : translate();
};
m.clipExtent = function(_) {
if (!arguments.length) return clipAuto ? null : clipExtent();
if (clipAuto = _ == null) {
var k = pi$3 * scale(),
t = translate();
_ = [[t[0] - k, t[1] - k], [t[0] + k, t[1] + k]];
}
clipExtent(_);
return m;
};
return m.clipExtent(null);
}
function tany(y) {
return tan((halfPi$2 + y) / 2);
}
function conicConformalRaw(y0, y1) {
var cy0 = cos(y0),
n = y0 === y1 ? sin(y0) : log$1(cy0 / cos(y1)) / log$1(tany(y1) / tany(y0)),
f = cy0 * pow$1(tany(y0), n) / n;
if (!n) return mercatorRaw;
function project(x, y) {
if (f > 0) { if (y < -halfPi$2 + epsilon$4) y = -halfPi$2 + epsilon$4; }
else { if (y > halfPi$2 - epsilon$4) y = halfPi$2 - epsilon$4; }
var r = f / pow$1(tany(y), n);
return [r * sin(n * x), f - r * cos(n * x)];
}
project.invert = function(x, y) {
var fy = f - y, r = sign$1(n) * sqrt$1(x * x + fy * fy);
return [atan2(x, fy) / n, 2 * atan(pow$1(f / r, 1 / n)) - halfPi$2];
};
return project;
}
function conicConformal() {
return conicProjection(conicConformalRaw)
.scale(109.5)
.parallels([30, 30]);
}
function equirectangularRaw(lambda, phi) {
return [lambda, phi];
}
equirectangularRaw.invert = equirectangularRaw;
function equirectangular() {
return projection(equirectangularRaw)
.scale(152.63);
}
function conicEquidistantRaw(y0, y1) {
var cy0 = cos(y0),
n = y0 === y1 ? sin(y0) : (cy0 - cos(y1)) / (y1 - y0),
g = cy0 / n + y0;
if (abs(n) < epsilon$4) return equirectangularRaw;
function project(x, y) {
var gy = g - y, nx = n * x;
return [gy * sin(nx), g - gy * cos(nx)];
}
project.invert = function(x, y) {
var gy = g - y;
return [atan2(x, gy) / n, g - sign$1(n) * sqrt$1(x * x + gy * gy)];
};
return project;
}
function conicEquidistant() {
return conicProjection(conicEquidistantRaw)
.scale(131.154)
.center([0, 13.9389]);
}
function gnomonicRaw(x, y) {
var cy = cos(y), k = cos(x) * cy;
return [cy * sin(x) / k, sin(y) / k];
}
gnomonicRaw.invert = azimuthalInvert(atan);
function gnomonic() {
return projection(gnomonicRaw)
.scale(144.049)
.clipAngle(60);
}
function orthographicRaw(x, y) {
return [cos(y) * sin(x), sin(y)];
}
orthographicRaw.invert = azimuthalInvert(asin$1);
function orthographic() {
return projection(orthographicRaw)
.scale(249.5)
.clipAngle(90 + epsilon$4);
}
function stereographicRaw(x, y) {
var cy = cos(y), k = 1 + cos(x) * cy;
return [cy * sin(x) / k, sin(y) / k];
}
stereographicRaw.invert = azimuthalInvert(function(z) {
return 2 * atan(z);
});
function stereographic() {
return projection(stereographicRaw)
.scale(250)
.clipAngle(142);
}
function transverseMercatorRaw(lambda, phi) {
return [log$1(tan((halfPi$2 + phi) / 2)), -lambda];
}
transverseMercatorRaw.invert = function(x, y) {
return [-y, 2 * atan(exp(x)) - halfPi$2];
};
function transverseMercator() {
var m = mercatorProjection(transverseMercatorRaw),
center = m.center,
rotate = m.rotate;
m.center = function(_) {
return arguments.length ? center([-_[1], _[0]]) : (_ = center(), [_[1], -_[0]]);
};
m.rotate = function(_) {
return arguments.length ? rotate([_[0], _[1], _.length > 2 ? _[2] + 90 : 90]) : (_ = rotate(), [_[0], _[1], _[2] - 90]);
};
return rotate([0, 0, 90])
.scale(159.155);
}
function attrsFunction(selection, map) {
return selection.each(function() {
var x = map.apply(this, arguments), s = select(this);
for (var name in x) s.attr(name, x[name]);
});
}
function attrsObject(selection, map) {
for (var name in map) selection.attr(name, map[name]);
return selection;
}
function selection_attrs(map) {
return (typeof map === "function" ? attrsFunction : attrsObject)(this, map);
}
function stylesFunction(selection, map, priority) {
return selection.each(function() {
var x = map.apply(this, arguments), s = select(this);
for (var name in x) s.style(name, x[name], priority);
});
}
function stylesObject(selection, map, priority) {
for (var name in map) selection.style(name, map[name], priority);
return selection;
}
function selection_styles(map, priority) {
return (typeof map === "function" ? stylesFunction : stylesObject)(this, map, priority == null ? "" : priority);
}
function propertiesFunction(selection, map) {
return selection.each(function() {
var x = map.apply(this, arguments), s = select(this);
for (var name in x) s.property(name, x[name]);
});
}
function propertiesObject(selection, map) {
for (var name in map) selection.property(name, map[name]);
return selection;
}
function selection_properties(map) {
return (typeof map === "function" ? propertiesFunction : propertiesObject)(this, map);
}
function attrsFunction$1(transition, map) {
return transition.each(function() {
var x = map.apply(this, arguments), t = select(this).transition(transition);
for (var name in x) t.attr(name, x[name]);
});
}
function attrsObject$1(transition, map) {
for (var name in map) transition.attr(name, map[name]);
return transition;
}
function transition_attrs(map) {
return (typeof map === "function" ? attrsFunction$1 : attrsObject$1)(this, map);
}
function stylesFunction$1(transition, map, priority) {
return transition.each(function() {
var x = map.apply(this, arguments), t = select(this).transition(transition);
for (var name in x) t.style(name, x[name], priority);
});
}
function stylesObject$1(transition, map, priority) {
for (var name in map) transition.style(name, map[name], priority);
return transition;
}
function transition_styles(map, priority) {
return (typeof map === "function" ? stylesFunction$1 : stylesObject$1)(this, map, priority == null ? "" : priority);
}
selection.prototype.attrs = selection_attrs;
selection.prototype.styles = selection_styles;
selection.prototype.properties = selection_properties;
transition.prototype.attrs = transition_attrs;
transition.prototype.styles = transition_styles;
function translateSelection(xy) {
return this.attr('transform', function(d,i) {
return 'translate('+[typeof xy == 'function' ? xy.call(this, d,i) : xy]+')';
});
};
function parseAttributes(name) {
if (typeof name === "string") {
var attr = {},
parts = name.split(/([\.#])/g), p;
name = parts.shift();
while ((p = parts.shift())) {
if (p == '.') attr['class'] = attr['class'] ? attr['class'] + ' ' + parts.shift() : parts.shift();
else if (p == '#') attr.id = parts.shift();
}
return {tag: name, attr: attr};
}
return name;
}
function append(name) {
var n = parseAttributes(name), s;
name = creator(n.tag);
s = this.select(function() {
return this.appendChild(name.apply(this, arguments));
});
//attrs not provided by default in v4
for (var key in n.attr) { s.attr(key, n.attr[key]) }
return s;
};
function selectAppend(name) {
var select = selector(name),
n = parseAttributes(name), s;
name = creator(n.tag);
s = this.select(function() {
return select.apply(this, arguments)
|| this.appendChild(name.apply(this, arguments));
});
//attrs not provided by default in v4
for (var key in n.attr) { s.attr(key, n.attr[key]) }
return s;
};
function tspans(lines, lh) {
return this.selectAll('tspan')
.data(lines).enter()
.append('tspan')
.text(function(d) { return d; })
.attr('x', 0)
.attr('dy', function(d, i) { return i ? lh || 15 : 0; });
};
function appendMany(data, name){
return this.selectAll(null).data(data).enter().append(name);
};
function at(name, value) {
if (typeof(name) == 'object'){
for (var key in name){
this.attr(key.replace(/([a-z\d])([A-Z])/g, '$1-$2').toLowerCase(), name[key])
}
return this
} else{
return arguments.length == 1 ? this.attr(name) : this.attr(name, value)
}
};
function f(){
var functions = arguments
//convert all string arguments into field accessors
var i = 0, l = functions.length
while (i < l) {
if (typeof(functions[i]) === 'string' || typeof(functions[i]) === 'number'){
functions[i] = (function(str){ return function(d){ return d[str] } })(functions[i])
}
i++
}
//return composition of functions
return function(d) {
var i=0, l = functions.length
while (i++ < l) d = functions[i-1].call(this, d)
return d
}
}
f.not = function(d){ return !d }
f.run = function(d){ return d() }
f.objToFn = function(obj, defaultVal){
if (arguments.length == 1) defaultVal = undefined
return function(str){
return typeof(obj[str]) !== undefined ? obj[str] : defaultVal }
}
function st(name, value) {
if (typeof(name) == 'object'){
for (var key in name){
addStyle(this, key, name[key])
}
return this
} else{
return arguments.length == 1 ? this.style(name) : addStyle(this, name, value)
}
function addStyle(sel, style, value){
var style = style.replace(/([a-z\d])([A-Z])/g, '$1-$2').toLowerCase()
var pxStyles = 'top left bottom right padding-top padding-left padding-bottom padding-right border-top b-width border-left-width border-botto-width m border-right-width margin-top margin-left margin-bottom margin-right font-size width height stroke-width line-height margin padding border max-width min-width'
if (~pxStyles.indexOf(style) ){
sel.style(style, typeof value == 'function' ? f(value, addPx) : addPx(value))
} else{
sel.style(style, value)
}
return sel
}
function addPx(d){ return d.match ? d : d + 'px' }
};
function wordwrap(line, maxCharactersPerLine) {
var w = line.split(' '),
lines = [],
words = [],
maxChars = maxCharactersPerLine || 40,
l = 0;
w.forEach(function(d) {
if (l+d.length > maxChars) {
lines.push(words.join(' '));
words.length = 0;
l = 0;
}
l += d.length;
words.push(d);
});
if (words.length) {
lines.push(words.join(' '));
}
return lines.filter(function(d){ return d != '' });
};
function ascendingKey(key) {
return typeof key == 'function' ? function (a, b) {
return key(a) < key(b) ? -1 : key(a) > key(b) ? 1 : key(a) >= key(b) ? 0 : NaN;
} : function (a, b) {
return a[key] < b[key] ? -1 : a[key] > b[key] ? 1 : a[key] >= b[key] ? 0 : NaN;
};
};
function descendingKey(key) {
return typeof key == 'function' ? function (a, b) {
return key(b) < key(a) ? -1 : key(b) > key(a) ? 1 : key(b) >= key(a) ? 0 : NaN;
} : function (a, b) {
return b[key] < a[key] ? -1 : b[key] > a[key] ? 1 : b[key] >= a[key] ? 0 : NaN;
};
};
function conventions(c){
c = c || {}
c.margin = c.margin || {top: 20, right: 20, bottom: 20, left: 20}
;['top', 'right', 'bottom', 'left'].forEach(function(d){
if (!c.margin[d] && c.margin[d] != 0) c.margin[d] = 20
})
c.width = c.width || c.totalWidth - c.margin.left - c.margin.right || 900
c.height = c.height || c.totalHeight - c.margin.top - c.margin.bottom || 460
c.totalWidth = c.width + c.margin.left + c.margin.right
c.totalHeight = c.height + c.margin.top + c.margin.bottom
c.parentSel = c.parentSel || select('body')
c.rootsvg = c.parentSel.append('svg')
c.svg = c.rootsvg
.attr('width', c.totalWidth)
.attr('height', c.totalHeight)
.append('g')
.attr('transform', 'translate(' + c.margin.left + ',' + c.margin.top + ')')
c.x = c.x || linear$2().range([0, c.width])
c.y = c.y || linear$2().range([c.height, 0])
c.xAxis = c.xAxis || axisBottom().scale(c.x)
c.yAxis = c.yAxis || axisLeft().scale(c.y)
c.drawAxis = function(){
c.svg.append('g')
.attr('class', 'x axis')
.attr('transform', 'translate(0,' + c.height + ')')
.call(c.xAxis);
c.svg.append('g')
.attr('class', 'y axis')
.call(c.yAxis);
}
return c
}
function attachTooltip(sel, tooltipSel, fieldFns){
if (!sel.size()) return
tooltipSel = tooltipSel || select('.tooltip')
sel
.on('mouseover.attachTooltip', ttDisplay)
.on('mousemove.attachTooltip', ttMove)
.on('mouseout.attachTooltip', ttHide)
.on('click.attachTooltip', function(d){ console.log(d) })
var d = sel.datum()
fieldFns = fieldFns || d3keys(d)
.filter(function(str){
return (typeof d[str] != 'object') && (d[str] != 'array')
})
.map(function(str){
return function(d){ return str + ': <b>' + d[str] + '</b>'} })
function ttDisplay(d){
tooltipSel
.classed('tooltip-hidden', false)
.html('')
.appendMany(fieldFns, 'div')
.html(function(fn){ return fn(d) })
select(this).classed('tooltipped', true)
}
function ttMove(d){
var tt = tooltipSel
if (!tt.size()) return
var e = exports.event,
x = e.clientX,
y = e.clientY,
n = tt.node(),
nBB = n.getBoundingClientRect(),
doctop = (window.scrollY)? window.scrollY : (document.documentElement && document.documentElement.scrollTop)? document.documentElement.scrollTop : document.body.scrollTop,
topPos = y+doctop-nBB.height-18;
tt.style('top', (topPos < 0 ? 18 + y : topPos)+'px');
tt.style('left', Math.min(Math.max(20, (x-nBB.width/2)), window.innerWidth - nBB.width - 20)+'px');
}
function ttHide(d){
tooltipSel.classed('tooltip-hidden', true);
selectAll('.tooltipped').classed('tooltipped', false)
}
}
function loadData(files, cb){
var q = queue()
files.forEach(function(d){
var type = d.split('.').reverse()[0]
var loadFn = {csv: csv$1, tsv: tsv$1, json: json}[type]
if (!loadFn) return cb(new Error('Invalid type', d))
q.defer(loadFn, d)
})
q.awaitAll(cb)
}
function nestBy(array, key){
return nest().key(key).entries(array).map(function(d){
d.values.key = d.key
return d.values
})
}
function round(n, p) {
return p ? Math.round(n * (p = Math.pow(10, p))) / p : Math.round(n);
};
selection.prototype.translate = translateSelection
selection.prototype.append = append
selection.prototype.selectAppend = selectAppend
selection.prototype.tspans = tspans
selection.prototype.appendMany = appendMany
selection.prototype.at = at
selection.prototype.st = st
selection.prototype.prop = selection.prototype.property
exports.bisect = bisectRight;
exports.bisectRight = bisectRight;
exports.bisectLeft = bisectLeft;
exports.ascending = ascending;
exports.bisector = bisector;
exports.descending = descending;
exports.deviation = deviation;
exports.extent = extent;
exports.histogram = histogram;
exports.thresholdFreedmanDiaconis = freedmanDiaconis;
exports.thresholdScott = scott;
exports.thresholdSturges = sturges;
exports.max = max;
exports.mean = mean;
exports.median = median;
exports.merge = merge;
exports.min = min;
exports.pairs = pairs;
exports.permute = permute;
exports.quantile = threshold;
exports.range = range;
exports.scan = scan;
exports.shuffle = shuffle;
exports.sum = sum;
exports.ticks = ticks;
exports.tickStep = tickStep;
exports.transpose = transpose;
exports.variance = variance;
exports.zip = zip;
exports.entries = entries;
exports.keys = d3keys;
exports.values = values;
exports.map = map$1;
exports.set = set;
exports.nest = nest;
exports.randomUniform = uniform;
exports.randomNormal = normal;
exports.randomLogNormal = logNormal;
exports.randomBates = bates;
exports.randomIrwinHall = irwinHall;
exports.randomExponential = exponential;
exports.easeLinear = linear;
exports.easeQuad = quadInOut;
exports.easeQuadIn = quadIn;
exports.easeQuadOut = quadOut;
exports.easeQuadInOut = quadInOut;
exports.easeCubic = easeCubicInOut;
exports.easeCubicIn = cubicIn;
exports.easeCubicOut = cubicOut;
exports.easeCubicInOut = easeCubicInOut;
exports.easePoly = polyInOut;
exports.easePolyIn = polyIn;
exports.easePolyOut = polyOut;
exports.easePolyInOut = polyInOut;
exports.easeSin = sinInOut;
exports.easeSinIn = sinIn;
exports.easeSinOut = sinOut;
exports.easeSinInOut = sinInOut;
exports.easeExp = expInOut;
exports.easeExpIn = expIn;
exports.easeExpOut = expOut;
exports.easeExpInOut = expInOut;
exports.easeCircle = circleInOut;
exports.easeCircleIn = circleIn;
exports.easeCircleOut = circleOut;
exports.easeCircleInOut = circleInOut;
exports.easeBounce = bounceOut;
exports.easeBounceIn = bounceIn;
exports.easeBounceOut = bounceOut;
exports.easeBounceInOut = bounceInOut;
exports.easeBack = backInOut;
exports.easeBackIn = backIn;
exports.easeBackOut = backOut;
exports.easeBackInOut = backInOut;
exports.easeElastic = elasticOut;
exports.easeElasticIn = elasticIn;
exports.easeElasticOut = elasticOut;
exports.easeElasticInOut = elasticInOut;
exports.polygonArea = area;
exports.polygonCentroid = centroid;
exports.polygonHull = hull;
exports.polygonContains = contains;
exports.polygonLength = length$1;
exports.path = path;
exports.quadtree = quadtree;
exports.queue = queue;
exports.arc = arc;
exports.area = area$1;
exports.line = line;
exports.pie = pie;
exports.radialArea = radialArea;
exports.radialLine = radialLine$1;
exports.symbol = symbol;
exports.symbols = symbols;
exports.symbolCircle = circle;
exports.symbolCross = cross$1;
exports.symbolDiamond = diamond;
exports.symbolSquare = square;
exports.symbolStar = star;
exports.symbolTriangle = triangle;
exports.symbolWye = wye;
exports.curveBasisClosed = basisClosed;
exports.curveBasisOpen = basisOpen;
exports.curveBasis = basis;
exports.curveBundle = bundle;
exports.curveCardinalClosed = cardinalClosed;
exports.curveCardinalOpen = cardinalOpen;
exports.curveCardinal = cardinal;
exports.curveCatmullRomClosed = catmullRomClosed;
exports.curveCatmullRomOpen = catmullRomOpen;
exports.curveCatmullRom = catmullRom;
exports.curveLinearClosed = linearClosed;
exports.curveLinear = curveLinear;
exports.curveMonotoneX = monotoneX;
exports.curveMonotoneY = monotoneY;
exports.curveNatural = natural;
exports.curveStep = step;
exports.curveStepAfter = stepAfter;
exports.curveStepBefore = stepBefore;
exports.stack = stack;
exports.stackOffsetExpand = expand;
exports.stackOffsetNone = none;
exports.stackOffsetSilhouette = silhouette;
exports.stackOffsetWiggle = wiggle;
exports.stackOrderAscending = ascending$1;
exports.stackOrderDescending = descending$2;
exports.stackOrderInsideOut = insideOut;
exports.stackOrderNone = none$1;
exports.stackOrderReverse = reverse;
exports.color = color;
exports.rgb = colorRgb;
exports.hsl = colorHsl;
exports.lab = lab;
exports.hcl = colorHcl;
exports.cubehelix = cubehelix;
exports.interpolate = interpolate;
exports.interpolateArray = array$1;
exports.interpolateNumber = interpolateNumber;
exports.interpolateObject = object;
exports.interpolateRound = interpolateRound;
exports.interpolateString = interpolateString;
exports.interpolateTransformCss = interpolateTransform$1;
exports.interpolateTransformSvg = interpolateTransform$2;
exports.interpolateZoom = interpolateZoom;
exports.interpolateRgb = interpolateRgb;
exports.interpolateRgbBasis = rgbBasis;
exports.interpolateRgbBasisClosed = rgbBasisClosed;
exports.interpolateHsl = hsl$1;
exports.interpolateHslLong = hslLong;
exports.interpolateLab = lab$1;
exports.interpolateHcl = hcl$1;
exports.interpolateHclLong = hclLong;
exports.interpolateCubehelix = cubehelix$2;
exports.interpolateCubehelixLong = interpolateCubehelixLong;
exports.interpolateBasis = basis$2;
exports.interpolateBasisClosed = basisClosed$1;
exports.quantize = quantize;
exports.dispatch = dispatch;
exports.dsvFormat = dsv;
exports.csvParse = csvParse;
exports.csvParseRows = csvParseRows;
exports.csvFormat = csvFormat;
exports.csvFormatRows = csvFormatRows;
exports.tsvParse = tsvParse;
exports.tsvParseRows = tsvParseRows;
exports.tsvFormat = tsvFormat;
exports.tsvFormatRows = tsvFormatRows;
exports.request = request;
exports.html = html;
exports.json = json;
exports.text = text;
exports.xml = xml;
exports.csv = csv$1;
exports.tsv = tsv$1;
exports.now = now;
exports.timer = timer;
exports.timerFlush = timerFlush;
exports.timeout = timeout$1;
exports.interval = interval$1;
exports.timeInterval = newInterval;
exports.timeMillisecond = millisecond;
exports.timeMilliseconds = milliseconds;
exports.timeSecond = second;
exports.timeSeconds = seconds;
exports.timeMinute = minute;
exports.timeMinutes = minutes;
exports.timeHour = hour;
exports.timeHours = hours;
exports.timeDay = day;
exports.timeDays = days;
exports.timeWeek = timeWeek;
exports.timeWeeks = sundays;
exports.timeSunday = timeWeek;
exports.timeSundays = sundays;
exports.timeMonday = timeMonday;
exports.timeMondays = mondays;
exports.timeTuesday = tuesday;
exports.timeTuesdays = tuesdays;
exports.timeWednesday = wednesday;
exports.timeWednesdays = wednesdays;
exports.timeThursday = thursday;
exports.timeThursdays = thursdays;
exports.timeFriday = friday;
exports.timeFridays = fridays;
exports.timeSaturday = saturday;
exports.timeSaturdays = saturdays;
exports.timeMonth = month;
exports.timeMonths = months;
exports.timeYear = year;
exports.timeYears = years;
exports.utcMillisecond = millisecond;
exports.utcMilliseconds = milliseconds;
exports.utcSecond = second;
exports.utcSeconds = seconds;
exports.utcMinute = utcMinute;
exports.utcMinutes = utcMinutes;
exports.utcHour = utcHour;
exports.utcHours = utcHours;
exports.utcDay = utcDay;
exports.utcDays = utcDays;
exports.utcWeek = utcWeek;
exports.utcWeeks = utcSundays;
exports.utcSunday = utcWeek;
exports.utcSundays = utcSundays;
exports.utcMonday = utcMonday;
exports.utcMondays = utcMondays;
exports.utcTuesday = utcTuesday;
exports.utcTuesdays = utcTuesdays;
exports.utcWednesday = utcWednesday;
exports.utcWednesdays = utcWednesdays;
exports.utcThursday = utcThursday;
exports.utcThursdays = utcThursdays;
exports.utcFriday = utcFriday;
exports.utcFridays = utcFridays;
exports.utcSaturday = utcSaturday;
exports.utcSaturdays = utcSaturdays;
exports.utcMonth = utcMonth;
exports.utcMonths = utcMonths;
exports.utcYear = utcYear;
exports.utcYears = utcYears;
exports.formatLocale = formatLocale;
exports.formatDefaultLocale = defaultLocale;
exports.formatSpecifier = formatSpecifier;
exports.precisionFixed = precisionFixed;
exports.precisionPrefix = precisionPrefix;
exports.precisionRound = precisionRound;
exports.isoFormat = formatIso;
exports.isoParse = parseIso;
exports.timeFormatLocale = formatLocale$1;
exports.timeFormatDefaultLocale = defaultLocale$1;
exports.scaleBand = band;
exports.scalePoint = point$4;
exports.scaleIdentity = identity$4;
exports.scaleLinear = linear$2;
exports.scaleLog = log;
exports.scaleOrdinal = ordinal;
exports.scaleImplicit = implicit;
exports.scalePow = pow;
exports.scaleSqrt = sqrt;
exports.scaleQuantile = quantile;
exports.scaleQuantize = quantize$1;
exports.scaleThreshold = threshold$1;
exports.scaleTime = time;
exports.scaleUtc = utcTime;
exports.schemeCategory10 = category10;
exports.schemeCategory20b = category20b;
exports.schemeCategory20c = category20c;
exports.schemeCategory20 = category20;
exports.scaleSequential = sequential;
exports.interpolateCubehelixDefault = cubehelix$3;
exports.interpolateRainbow = rainbow$1;
exports.interpolateWarm = warm;
exports.interpolateCool = cool;
exports.interpolateViridis = viridis;
exports.interpolateMagma = magma;
exports.interpolateInferno = inferno;
exports.interpolatePlasma = plasma;
exports.creator = creator;
exports.customEvent = customEvent;
exports.local = local;
exports.matcher = matcher$1;
exports.mouse = mouse;
exports.namespace = namespace;
exports.namespaces = namespaces;
exports.select = select;
exports.selectAll = selectAll;
exports.selection = selection;
exports.selector = selector;
exports.selectorAll = selectorAll;
exports.touch = touch;
exports.touches = touches;
exports.window = window$1;
exports.active = active;
exports.interrupt = interrupt;
exports.transition = transition;
exports.axisTop = axisTop;
exports.axisRight = axisRight;
exports.axisBottom = axisBottom;
exports.axisLeft = axisLeft;
exports.cluster = cluster;
exports.hierarchy = hierarchy;
exports.pack = index;
exports.packSiblings = siblings;
exports.packEnclose = enclose;
exports.partition = partition;
exports.stratify = stratify;
exports.tree = tree;
exports.treemap = index$1;
exports.treemapBinary = binary;
exports.treemapDice = treemapDice;
exports.treemapSlice = treemapSlice;
exports.treemapSliceDice = sliceDice;
exports.treemapSquarify = squarify;
exports.treemapResquarify = resquarify;
exports.forceCenter = center$1;
exports.forceCollide = collide;
exports.forceLink = link;
exports.forceManyBody = manyBody;
exports.forceSimulation = simulation;
exports.forceX = x$3;
exports.forceY = y$3;
exports.drag = drag;
exports.dragDisable = dragDisable;
exports.dragEnable = dragEnable;
exports.voronoi = voronoi;
exports.zoom = zoom;
exports.zoomIdentity = identity$6;
exports.zoomTransform = transform;
exports.brush = brush;
exports.brushX = brushX;
exports.brushY = brushY;
exports.brushSelection = brushSelection;
exports.geoArea = area$2;
exports.geoBounds = bounds;
exports.geoCentroid = centroid$1;
exports.geoCircle = circle$1;
exports.geoClipExtent = extent$1;
exports.geoDistance = distance;
exports.geoGraticule = graticule;
exports.geoInterpolate = interpolate$2;
exports.geoLength = length$2;
exports.geoPath = index$3;
exports.geoAlbers = albers;
exports.geoAlbersUsa = albersUsa;
exports.geoAzimuthalEqualArea = azimuthalEqualArea;
exports.geoAzimuthalEquidistant = azimuthalEquidistant;
exports.geoConicConformal = conicConformal;
exports.geoConicEqualArea = conicEqualArea;
exports.geoConicEquidistant = conicEquidistant;
exports.geoEquirectangular = equirectangular;
exports.geoGnomonic = gnomonic;
exports.geoProjection = projection;
exports.geoProjectionMutator = projectionMutator;
exports.geoMercator = mercator;
exports.geoOrthographic = orthographic;
exports.geoStereographic = stereographic;
exports.geoTransverseMercator = transverseMercator;
exports.geoRotation = rotation;
exports.geoStream = geoStream;
exports.geoTransform = transform$1;
exports.wordwrap = wordwrap;
exports.parseAttributes = parseAttributes;
exports.f = f;
exports.ascendingKey = ascendingKey;
exports.descendingKey = descendingKey;
exports.conventions = conventions;
exports.attachTooltip = attachTooltip;
exports.loadData = loadData;
exports.nestBy = nestBy;
exports.round = round;
Object.defineProperty(exports, '__esModule', { value: true });
}));
/**
* @license
* lodash lodash.com/license | Underscore.js 1.8.3 underscorejs.org/LICENSE
*/
;(function(){function t(t,n){return t.set(n[0],n[1]),t}function n(t,n){return t.add(n),t}function r(t,n,r){switch(r.length){case 0:return t.call(n);case 1:return t.call(n,r[0]);case 2:return t.call(n,r[0],r[1]);case 3:return t.call(n,r[0],r[1],r[2])}return t.apply(n,r)}function e(t,n,r,e){for(var u=-1,i=t?t.length:0;++u<i;){var o=t[u];n(e,o,r(o),t)}return e}function u(t,n){for(var r=-1,e=t?t.length:0;++r<e&&false!==n(t[r],r,t););return t}function i(t,n){for(var r=t?t.length:0;r--&&false!==n(t[r],r,t););
return t}function o(t,n){for(var r=-1,e=t?t.length:0;++r<e;)if(!n(t[r],r,t))return false;return true}function f(t,n){for(var r=-1,e=t?t.length:0,u=0,i=[];++r<e;){var o=t[r];n(o,r,t)&&(i[u++]=o)}return i}function c(t,n){return!(!t||!t.length)&&-1<d(t,n,0)}function a(t,n,r){for(var e=-1,u=t?t.length:0;++e<u;)if(r(n,t[e]))return true;return false}function l(t,n){for(var r=-1,e=t?t.length:0,u=Array(e);++r<e;)u[r]=n(t[r],r,t);return u}function s(t,n){for(var r=-1,e=n.length,u=t.length;++r<e;)t[u+r]=n[r];return t}function h(t,n,r,e){
var u=-1,i=t?t.length:0;for(e&&i&&(r=t[++u]);++u<i;)r=n(r,t[u],u,t);return r}function p(t,n,r,e){var u=t?t.length:0;for(e&&u&&(r=t[--u]);u--;)r=n(r,t[u],u,t);return r}function _(t,n){for(var r=-1,e=t?t.length:0;++r<e;)if(n(t[r],r,t))return true;return false}function v(t,n,r){var e;return r(t,function(t,r,u){if(n(t,r,u))return e=r,false}),e}function g(t,n,r,e){var u=t.length;for(r+=e?1:-1;e?r--:++r<u;)if(n(t[r],r,t))return r;return-1}function d(t,n,r){if(n===n)t:{--r;for(var e=t.length;++r<e;)if(t[r]===n){t=r;
break t}t=-1}else t=g(t,b,r);return t}function y(t,n,r,e){--r;for(var u=t.length;++r<u;)if(e(t[r],n))return r;return-1}function b(t){return t!==t}function x(t,n){var r=t?t.length:0;return r?k(t,n)/r:P}function j(t){return function(n){return null==n?F:n[t]}}function w(t){return function(n){return null==t?F:t[n]}}function m(t,n,r,e,u){return u(t,function(t,u,i){r=e?(e=false,t):n(r,t,u,i)}),r}function A(t,n){var r=t.length;for(t.sort(n);r--;)t[r]=t[r].c;return t}function k(t,n){for(var r,e=-1,u=t.length;++e<u;){
var i=n(t[e]);i!==F&&(r=r===F?i:r+i)}return r}function E(t,n){for(var r=-1,e=Array(t);++r<t;)e[r]=n(r);return e}function O(t,n){return l(n,function(n){return[n,t[n]]})}function S(t){return function(n){return t(n)}}function I(t,n){return l(n,function(n){return t[n]})}function R(t,n){return t.has(n)}function z(t,n){for(var r=-1,e=t.length;++r<e&&-1<d(n,t[r],0););return r}function W(t,n){for(var r=t.length;r--&&-1<d(n,t[r],0););return r}function B(t){return"\\"+Dt[t]}function L(t){var n=-1,r=Array(t.size);
return t.forEach(function(t,e){r[++n]=[e,t]}),r}function U(t,n){return function(r){return t(n(r))}}function C(t,n){for(var r=-1,e=t.length,u=0,i=[];++r<e;){var o=t[r];o!==n&&"__lodash_placeholder__"!==o||(t[r]="__lodash_placeholder__",i[u++]=r)}return i}function M(t){var n=-1,r=Array(t.size);return t.forEach(function(t){r[++n]=t}),r}function D(t){var n=-1,r=Array(t.size);return t.forEach(function(t){r[++n]=[t,t]}),r}function T(t){if(Wt.test(t)){for(var n=Rt.lastIndex=0;Rt.test(t);)++n;t=n}else t=tn(t);
return t}function $(t){return Wt.test(t)?t.match(Rt)||[]:t.split("")}var F,N=1/0,P=NaN,Z=[["ary",128],["bind",1],["bindKey",2],["curry",8],["curryRight",16],["flip",512],["partial",32],["partialRight",64],["rearg",256]],q=/\b__p\+='';/g,V=/\b(__p\+=)''\+/g,K=/(__e\(.*?\)|\b__t\))\+'';/g,G=/&(?:amp|lt|gt|quot|#39);/g,J=/[&<>"']/g,Y=RegExp(G.source),H=RegExp(J.source),Q=/<%-([\s\S]+?)%>/g,X=/<%([\s\S]+?)%>/g,tt=/<%=([\s\S]+?)%>/g,nt=/\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,rt=/^\w*$/,et=/^\./,ut=/[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g,it=/[\\^$.*+?()[\]{}|]/g,ot=RegExp(it.source),ft=/^\s+|\s+$/g,ct=/^\s+/,at=/\s+$/,lt=/\{(?:\n\/\* \[wrapped with .+\] \*\/)?\n?/,st=/\{\n\/\* \[wrapped with (.+)\] \*/,ht=/,? & /,pt=/[^\x00-\x2f\x3a-\x40\x5b-\x60\x7b-\x7f]+/g,_t=/\\(\\)?/g,vt=/\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g,gt=/\w*$/,dt=/^[-+]0x[0-9a-f]+$/i,yt=/^0b[01]+$/i,bt=/^\[object .+?Constructor\]$/,xt=/^0o[0-7]+$/i,jt=/^(?:0|[1-9]\d*)$/,wt=/[\xc0-\xd6\xd8-\xf6\xf8-\xff\u0100-\u017f]/g,mt=/($^)/,At=/['\n\r\u2028\u2029\\]/g,kt="[\\ufe0e\\ufe0f]?(?:[\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0]|\\ud83c[\\udffb-\\udfff])?(?:\\u200d(?:[^\\ud800-\\udfff]|(?:\\ud83c[\\udde6-\\uddff]){2}|[\\ud800-\\udbff][\\udc00-\\udfff])[\\ufe0e\\ufe0f]?(?:[\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0]|\\ud83c[\\udffb-\\udfff])?)*",Et="(?:[\\u2700-\\u27bf]|(?:\\ud83c[\\udde6-\\uddff]){2}|[\\ud800-\\udbff][\\udc00-\\udfff])"+kt,Ot="(?:[^\\ud800-\\udfff][\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0]?|[\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0]|(?:\\ud83c[\\udde6-\\uddff]){2}|[\\ud800-\\udbff][\\udc00-\\udfff]|[\\ud800-\\udfff])",St=RegExp("['\u2019]","g"),It=RegExp("[\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0]","g"),Rt=RegExp("\\ud83c[\\udffb-\\udfff](?=\\ud83c[\\udffb-\\udfff])|"+Ot+kt,"g"),zt=RegExp(["[A-Z\\xc0-\\xd6\\xd8-\\xde]?[a-z\\xdf-\\xf6\\xf8-\\xff]+(?:['\u2019](?:d|ll|m|re|s|t|ve))?(?=[\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000]|[A-Z\\xc0-\\xd6\\xd8-\\xde]|$)|(?:[A-Z\\xc0-\\xd6\\xd8-\\xde]|[^\\ud800-\\udfff\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000\\d+\\u2700-\\u27bfa-z\\xdf-\\xf6\\xf8-\\xffA-Z\\xc0-\\xd6\\xd8-\\xde])+(?:['\u2019](?:D|LL|M|RE|S|T|VE))?(?=[\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000]|[A-Z\\xc0-\\xd6\\xd8-\\xde](?:[a-z\\xdf-\\xf6\\xf8-\\xff]|[^\\ud800-\\udfff\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000\\d+\\u2700-\\u27bfa-z\\xdf-\\xf6\\xf8-\\xffA-Z\\xc0-\\xd6\\xd8-\\xde])|$)|[A-Z\\xc0-\\xd6\\xd8-\\xde]?(?:[a-z\\xdf-\\xf6\\xf8-\\xff]|[^\\ud800-\\udfff\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000\\d+\\u2700-\\u27bfa-z\\xdf-\\xf6\\xf8-\\xffA-Z\\xc0-\\xd6\\xd8-\\xde])+(?:['\u2019](?:d|ll|m|re|s|t|ve))?|[A-Z\\xc0-\\xd6\\xd8-\\xde]+(?:['\u2019](?:D|LL|M|RE|S|T|VE))?|\\d+",Et].join("|"),"g"),Wt=RegExp("[\\u200d\\ud800-\\udfff\\u0300-\\u036f\\ufe20-\\ufe23\\u20d0-\\u20f0\\ufe0e\\ufe0f]"),Bt=/[a-z][A-Z]|[A-Z]{2,}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/,Lt="Array Buffer DataView Date Error Float32Array Float64Array Function Int8Array Int16Array Int32Array Map Math Object Promise RegExp Set String Symbol TypeError Uint8Array Uint8ClampedArray Uint16Array Uint32Array WeakMap _ clearTimeout isFinite parseInt setTimeout".split(" "),Ut={};
Ut["[object Float32Array]"]=Ut["[object Float64Array]"]=Ut["[object Int8Array]"]=Ut["[object Int16Array]"]=Ut["[object Int32Array]"]=Ut["[object Uint8Array]"]=Ut["[object Uint8ClampedArray]"]=Ut["[object Uint16Array]"]=Ut["[object Uint32Array]"]=true,Ut["[object Arguments]"]=Ut["[object Array]"]=Ut["[object ArrayBuffer]"]=Ut["[object Boolean]"]=Ut["[object DataView]"]=Ut["[object Date]"]=Ut["[object Error]"]=Ut["[object Function]"]=Ut["[object Map]"]=Ut["[object Number]"]=Ut["[object Object]"]=Ut["[object RegExp]"]=Ut["[object Set]"]=Ut["[object String]"]=Ut["[object WeakMap]"]=false;
var Ct={};Ct["[object Arguments]"]=Ct["[object Array]"]=Ct["[object ArrayBuffer]"]=Ct["[object DataView]"]=Ct["[object Boolean]"]=Ct["[object Date]"]=Ct["[object Float32Array]"]=Ct["[object Float64Array]"]=Ct["[object Int8Array]"]=Ct["[object Int16Array]"]=Ct["[object Int32Array]"]=Ct["[object Map]"]=Ct["[object Number]"]=Ct["[object Object]"]=Ct["[object RegExp]"]=Ct["[object Set]"]=Ct["[object String]"]=Ct["[object Symbol]"]=Ct["[object Uint8Array]"]=Ct["[object Uint8ClampedArray]"]=Ct["[object Uint16Array]"]=Ct["[object Uint32Array]"]=true,
Ct["[object Error]"]=Ct["[object Function]"]=Ct["[object WeakMap]"]=false;var Mt,Dt={"\\":"\\","'":"'","\n":"n","\r":"r","\u2028":"u2028","\u2029":"u2029"},Tt=parseFloat,$t=parseInt,Ft=typeof global=="object"&&global&&global.Object===Object&&global,Nt=typeof self=="object"&&self&&self.Object===Object&&self,Pt=Ft||Nt||Function("return this")(),Zt=typeof exports=="object"&&exports&&!exports.nodeType&&exports,qt=Zt&&typeof module=="object"&&module&&!module.nodeType&&module,Vt=qt&&qt.exports===Zt,Kt=Vt&&Ft.h;
t:{try{Mt=Kt&&Kt.g("util");break t}catch(t){}Mt=void 0}var Gt=Mt&&Mt.isArrayBuffer,Jt=Mt&&Mt.isDate,Yt=Mt&&Mt.isMap,Ht=Mt&&Mt.isRegExp,Qt=Mt&&Mt.isSet,Xt=Mt&&Mt.isTypedArray,tn=j("length"),nn=w({"\xc0":"A","\xc1":"A","\xc2":"A","\xc3":"A","\xc4":"A","\xc5":"A","\xe0":"a","\xe1":"a","\xe2":"a","\xe3":"a","\xe4":"a","\xe5":"a","\xc7":"C","\xe7":"c","\xd0":"D","\xf0":"d","\xc8":"E","\xc9":"E","\xca":"E","\xcb":"E","\xe8":"e","\xe9":"e","\xea":"e","\xeb":"e","\xcc":"I","\xcd":"I","\xce":"I","\xcf":"I",
"\xec":"i","\xed":"i","\xee":"i","\xef":"i","\xd1":"N","\xf1":"n","\xd2":"O","\xd3":"O","\xd4":"O","\xd5":"O","\xd6":"O","\xd8":"O","\xf2":"o","\xf3":"o","\xf4":"o","\xf5":"o","\xf6":"o","\xf8":"o","\xd9":"U","\xda":"U","\xdb":"U","\xdc":"U","\xf9":"u","\xfa":"u","\xfb":"u","\xfc":"u","\xdd":"Y","\xfd":"y","\xff":"y","\xc6":"Ae","\xe6":"ae","\xde":"Th","\xfe":"th","\xdf":"ss","\u0100":"A","\u0102":"A","\u0104":"A","\u0101":"a","\u0103":"a","\u0105":"a","\u0106":"C","\u0108":"C","\u010a":"C","\u010c":"C",
"\u0107":"c","\u0109":"c","\u010b":"c","\u010d":"c","\u010e":"D","\u0110":"D","\u010f":"d","\u0111":"d","\u0112":"E","\u0114":"E","\u0116":"E","\u0118":"E","\u011a":"E","\u0113":"e","\u0115":"e","\u0117":"e","\u0119":"e","\u011b":"e","\u011c":"G","\u011e":"G","\u0120":"G","\u0122":"G","\u011d":"g","\u011f":"g","\u0121":"g","\u0123":"g","\u0124":"H","\u0126":"H","\u0125":"h","\u0127":"h","\u0128":"I","\u012a":"I","\u012c":"I","\u012e":"I","\u0130":"I","\u0129":"i","\u012b":"i","\u012d":"i","\u012f":"i",
"\u0131":"i","\u0134":"J","\u0135":"j","\u0136":"K","\u0137":"k","\u0138":"k","\u0139":"L","\u013b":"L","\u013d":"L","\u013f":"L","\u0141":"L","\u013a":"l","\u013c":"l","\u013e":"l","\u0140":"l","\u0142":"l","\u0143":"N","\u0145":"N","\u0147":"N","\u014a":"N","\u0144":"n","\u0146":"n","\u0148":"n","\u014b":"n","\u014c":"O","\u014e":"O","\u0150":"O","\u014d":"o","\u014f":"o","\u0151":"o","\u0154":"R","\u0156":"R","\u0158":"R","\u0155":"r","\u0157":"r","\u0159":"r","\u015a":"S","\u015c":"S","\u015e":"S",
"\u0160":"S","\u015b":"s","\u015d":"s","\u015f":"s","\u0161":"s","\u0162":"T","\u0164":"T","\u0166":"T","\u0163":"t","\u0165":"t","\u0167":"t","\u0168":"U","\u016a":"U","\u016c":"U","\u016e":"U","\u0170":"U","\u0172":"U","\u0169":"u","\u016b":"u","\u016d":"u","\u016f":"u","\u0171":"u","\u0173":"u","\u0174":"W","\u0175":"w","\u0176":"Y","\u0177":"y","\u0178":"Y","\u0179":"Z","\u017b":"Z","\u017d":"Z","\u017a":"z","\u017c":"z","\u017e":"z","\u0132":"IJ","\u0133":"ij","\u0152":"Oe","\u0153":"oe","\u0149":"'n",
"\u017f":"s"}),rn=w({"&":"&amp;","<":"&lt;",">":"&gt;",'"':"&quot;","'":"&#39;"}),en=w({"&amp;":"&","&lt;":"<","&gt;":">","&quot;":'"',"&#39;":"'"}),un=function w(kt){function Et(t){return fi.call(t)}function Ot(t){if(vu(t)&&!tf(t)&&!(t instanceof Dt)){if(t instanceof Mt)return t;if(ui.call(t,"__wrapped__"))return Me(t)}return new Mt(t)}function Rt(){}function Mt(t,n){this.__wrapped__=t,this.__actions__=[],this.__chain__=!!n,this.__index__=0,this.__values__=F}function Dt(t){this.__wrapped__=t,this.__actions__=[],
this.__dir__=1,this.__filtered__=false,this.__iteratees__=[],this.__takeCount__=4294967295,this.__views__=[]}function Ft(t){var n=-1,r=t?t.length:0;for(this.clear();++n<r;){var e=t[n];this.set(e[0],e[1])}}function Nt(t){var n=-1,r=t?t.length:0;for(this.clear();++n<r;){var e=t[n];this.set(e[0],e[1])}}function Zt(t){var n=-1,r=t?t.length:0;for(this.clear();++n<r;){var e=t[n];this.set(e[0],e[1])}}function qt(t){var n=-1,r=t?t.length:0;for(this.__data__=new Zt;++n<r;)this.add(t[n])}function Kt(t){this.size=(this.__data__=new Nt(t)).size;
}function tn(t,n){var r,e=tf(t)||fu(t)?E(t.length,Hu):[],u=e.length,i=!!u;for(r in t)!n&&!ui.call(t,r)||i&&("length"==r||we(r,u))||e.push(r);return e}function on(t){var n=t.length;return n?t[cr(0,n-1)]:F}function fn(t,n){return Be(Cr(t),n)}function cn(t){return Be(Cr(t))}function an(t,n,r,e){return t===F||ou(t,ti[r])&&!ui.call(e,r)?n:t}function ln(t,n,r){(r===F||ou(t[n],r))&&(r!==F||n in t)||vn(t,n,r)}function sn(t,n,r){var e=t[n];ui.call(t,n)&&ou(e,r)&&(r!==F||n in t)||vn(t,n,r)}function hn(t,n){
for(var r=t.length;r--;)if(ou(t[r][0],n))return r;return-1}function pn(t,n,r,e){return to(t,function(t,u,i){n(e,t,r(t),i)}),e}function _n(t,n){return t&&Mr(n,Iu(n),t)}function vn(t,n,r){"__proto__"==n&&xi?xi(t,n,{configurable:true,enumerable:true,value:r,writable:true}):t[n]=r}function gn(t,n){for(var r=-1,e=null==t,u=n.length,i=Zu(u);++r<u;)i[r]=e?F:Ou(t,n[r]);return i}function dn(t,n,r){return t===t&&(r!==F&&(t=t<=r?t:r),n!==F&&(t=t>=n?t:n)),t}function yn(t,n,r,e,i,o,f){var c;if(e&&(c=o?e(t,i,o,f):e(t)),
c!==F)return c;if(!_u(t))return t;if(i=tf(t)){if(c=ye(t),!n)return Cr(t,c)}else{var a=Et(t),l="[object Function]"==a||"[object GeneratorFunction]"==a;if(rf(t))return Rr(t,n);if("[object Object]"==a||"[object Arguments]"==a||l&&!o){if(c=be(l?{}:t),!n)return Dr(t,_n(c,t))}else{if(!Ct[a])return o?t:{};c=xe(t,a,yn,n)}}if(f||(f=new Kt),o=f.get(t))return o;if(f.set(t,c),!i)var s=r?zn(t,Iu,ao):Iu(t);return u(s||t,function(u,i){s&&(i=u,u=t[i]),sn(c,i,yn(u,n,r,e,i,t,f))}),c}function bn(t){var n=Iu(t);return function(r){
return xn(r,t,n)}}function xn(t,n,r){var e=r.length;if(null==t)return!e;for(t=Ju(t);e--;){var u=r[e],i=n[u],o=t[u];if(o===F&&!(u in t)||!i(o))return false}return true}function jn(t,n,r){if(typeof t!="function")throw new Qu("Expected a function");return po(function(){t.apply(F,r)},n)}function wn(t,n,r,e){var u=-1,i=c,o=true,f=t.length,s=[],h=n.length;if(!f)return s;r&&(n=l(n,S(r))),e?(i=a,o=false):200<=n.length&&(i=R,o=false,n=new qt(n));t:for(;++u<f;){var p=t[u],_=r?r(p):p,p=e||0!==p?p:0;if(o&&_===_){for(var v=h;v--;)if(n[v]===_)continue t;
s.push(p)}else i(n,_,e)||s.push(p)}return s}function mn(t,n){var r=true;return to(t,function(t,e,u){return r=!!n(t,e,u)}),r}function An(t,n,r){for(var e=-1,u=t.length;++e<u;){var i=t[e],o=n(i);if(null!=o&&(f===F?o===o&&!bu(o):r(o,f)))var f=o,c=i}return c}function kn(t,n){var r=[];return to(t,function(t,e,u){n(t,e,u)&&r.push(t)}),r}function En(t,n,r,e,u){var i=-1,o=t.length;for(r||(r=je),u||(u=[]);++i<o;){var f=t[i];0<n&&r(f)?1<n?En(f,n-1,r,e,u):s(u,f):e||(u[u.length]=f)}return u}function On(t,n){return t&&ro(t,n,Iu);
}function Sn(t,n){return t&&eo(t,n,Iu)}function In(t,n){return f(n,function(n){return su(t[n])})}function Rn(t,n){n=Ae(n,t)?[n]:Sr(n);for(var r=0,e=n.length;null!=t&&r<e;)t=t[Le(n[r++])];return r&&r==e?t:F}function zn(t,n,r){return n=n(t),tf(t)?n:s(n,r(t))}function Wn(t,n){return t>n}function Bn(t,n){return null!=t&&ui.call(t,n)}function Ln(t,n){return null!=t&&n in Ju(t)}function Un(t,n,r){for(var e=r?a:c,u=t[0].length,i=t.length,o=i,f=Zu(i),s=1/0,h=[];o--;){var p=t[o];o&&n&&(p=l(p,S(n))),s=Wi(p.length,s),
f[o]=!r&&(n||120<=u&&120<=p.length)?new qt(o&&p):F}var p=t[0],_=-1,v=f[0];t:for(;++_<u&&h.length<s;){var g=p[_],d=n?n(g):g,g=r||0!==g?g:0;if(v?!R(v,d):!e(h,d,r)){for(o=i;--o;){var y=f[o];if(y?!R(y,d):!e(t[o],d,r))continue t}v&&v.push(d),h.push(g)}}return h}function Cn(t,n,r){var e={};return On(t,function(t,u,i){n(e,r(t),u,i)}),e}function Mn(t,n,e){return Ae(n,t)||(n=Sr(n),t=Re(t,n),n=Ne(n)),n=null==t?t:t[Le(n)],null==n?F:r(n,t,e)}function Dn(t){return vu(t)&&"[object ArrayBuffer]"==fi.call(t)}function Tn(t){
return vu(t)&&"[object Date]"==fi.call(t)}function $n(t,n,r,e,u){if(t===n)n=true;else if(null==t||null==n||!_u(t)&&!vu(n))n=t!==t&&n!==n;else t:{var i=tf(t),o=tf(n),f="[object Array]",c="[object Array]";i||(f=Et(t),f="[object Arguments]"==f?"[object Object]":f),o||(c=Et(n),c="[object Arguments]"==c?"[object Object]":c);var a="[object Object]"==f,o="[object Object]"==c;if((c=f==c)&&rf(t)){if(!rf(n)){n=false;break t}i=true,a=false}if(c&&!a)u||(u=new Kt),n=i||cf(t)?ce(t,n,$n,r,e,u):ae(t,n,f,$n,r,e,u);else{if(!(2&e)&&(i=a&&ui.call(t,"__wrapped__"),
f=o&&ui.call(n,"__wrapped__"),i||f)){t=i?t.value():t,n=f?n.value():n,u||(u=new Kt),n=$n(t,n,r,e,u);break t}if(c)n:if(u||(u=new Kt),i=2&e,f=Iu(t),o=f.length,c=Iu(n).length,o==c||i){for(a=o;a--;){var l=f[a];if(!(i?l in n:ui.call(n,l))){n=false;break n}}if((c=u.get(t))&&u.get(n))n=c==n;else{c=true,u.set(t,n),u.set(n,t);for(var s=i;++a<o;){var l=f[a],h=t[l],p=n[l];if(r)var _=i?r(p,h,l,n,t,u):r(h,p,l,t,n,u);if(_===F?h!==p&&!$n(h,p,r,e,u):!_){c=false;break}s||(s="constructor"==l)}c&&!s&&(r=t.constructor,e=n.constructor,
r!=e&&"constructor"in t&&"constructor"in n&&!(typeof r=="function"&&r instanceof r&&typeof e=="function"&&e instanceof e)&&(c=false)),u.delete(t),u.delete(n),n=c}}else n=false;else n=false}}return n}function Fn(t){return vu(t)&&"[object Map]"==Et(t)}function Nn(t,n,r,e){var u=r.length,i=u,o=!e;if(null==t)return!i;for(t=Ju(t);u--;){var f=r[u];if(o&&f[2]?f[1]!==t[f[0]]:!(f[0]in t))return false}for(;++u<i;){var f=r[u],c=f[0],a=t[c],l=f[1];if(o&&f[2]){if(a===F&&!(c in t))return false}else{if(f=new Kt,e)var s=e(a,l,c,t,n,f);
if(s===F?!$n(l,a,e,3,f):!s)return false}}return true}function Pn(t){return!(!_u(t)||ri&&ri in t)&&(su(t)?ai:bt).test(Ue(t))}function Zn(t){return _u(t)&&"[object RegExp]"==fi.call(t)}function qn(t){return vu(t)&&"[object Set]"==Et(t)}function Vn(t){return vu(t)&&pu(t.length)&&!!Ut[fi.call(t)]}function Kn(t){return typeof t=="function"?t:null==t?Mu:typeof t=="object"?tf(t)?Xn(t[0],t[1]):Qn(t):Fu(t)}function Gn(t){if(!Ee(t))return Ri(t);var n,r=[];for(n in Ju(t))ui.call(t,n)&&"constructor"!=n&&r.push(n);return r;
}function Jn(t){if(!_u(t)){var n=[];if(null!=t)for(var r in Ju(t))n.push(r);return n}r=Ee(t);var e=[];for(n in t)("constructor"!=n||!r&&ui.call(t,n))&&e.push(n);return e}function Yn(t,n){return t<n}function Hn(t,n){var r=-1,e=cu(t)?Zu(t.length):[];return to(t,function(t,u,i){e[++r]=n(t,u,i)}),e}function Qn(t){var n=ve(t);return 1==n.length&&n[0][2]?Oe(n[0][0],n[0][1]):function(r){return r===t||Nn(r,t,n)}}function Xn(t,n){return Ae(t)&&n===n&&!_u(n)?Oe(Le(t),n):function(r){var e=Ou(r,t);return e===F&&e===n?Su(r,t):$n(n,e,F,3);
}}function tr(t,n,r,e,i){if(t!==n){if(!tf(n)&&!cf(n))var o=Jn(n);u(o||n,function(u,f){if(o&&(f=u,u=n[f]),_u(u)){i||(i=new Kt);var c=f,a=i,l=t[c],s=n[c],h=a.get(s);if(h)ln(t,c,h);else{var h=e?e(l,s,c+"",t,n,a):F,p=h===F;if(p){var _=tf(s),v=!_&&cf(s),h=s;_||v?tf(l)?h=l:au(l)?h=Cr(l):v?(p=false,h=Wr(s,true)):h=[]:du(s)||fu(s)?(h=l,fu(l)?h=ku(l):(!_u(l)||r&&su(l))&&(h=be(s))):p=false}p&&(a.set(s,h),tr(h,s,r,e,a),a.delete(s)),ln(t,c,h)}}else c=e?e(t[f],u,f+"",t,n,i):F,c===F&&(c=u),ln(t,f,c)})}}function nr(t,n){
var r=t.length;if(r)return n+=0>n?r:0,we(n,r)?t[n]:F}function rr(t,n,r){var e=-1;return n=l(n.length?n:[Mu],S(pe())),t=Hn(t,function(t){return{a:l(n,function(n){return n(t)}),b:++e,c:t}}),A(t,function(t,n){var e;t:{e=-1;for(var u=t.a,i=n.a,o=u.length,f=r.length;++e<o;){var c=Br(u[e],i[e]);if(c){e=e>=f?c:c*("desc"==r[e]?-1:1);break t}}e=t.b-n.b}return e})}function er(t,n){return t=Ju(t),ur(t,n,function(n,r){return r in t})}function ur(t,n,r){for(var e=-1,u=n.length,i={};++e<u;){var o=n[e],f=t[o];r(f,o)&&vn(i,o,f);
}return i}function ir(t){return function(n){return Rn(n,t)}}function or(t,n,r,e){var u=e?y:d,i=-1,o=n.length,f=t;for(t===n&&(n=Cr(n)),r&&(f=l(t,S(r)));++i<o;)for(var c=0,a=n[i],a=r?r(a):a;-1<(c=u(f,a,c,e));)f!==t&&yi.call(f,c,1),yi.call(t,c,1);return t}function fr(t,n){for(var r=t?n.length:0,e=r-1;r--;){var u=n[r];if(r==e||u!==i){var i=u;if(we(u))yi.call(t,u,1);else if(Ae(u,t))delete t[Le(u)];else{var u=Sr(u),o=Re(t,u);null!=o&&delete o[Le(Ne(u))]}}}}function cr(t,n){return t+ki(Ui()*(n-t+1))}function ar(t,n){
var r="";if(!t||1>n||9007199254740991<n)return r;do n%2&&(r+=t),(n=ki(n/2))&&(t+=t);while(n);return r}function lr(t,n){return _o(Ie(t,n,Mu),t+"")}function sr(t){return on(Wu(t))}function hr(t,n){return Be(Wu(t),n)}function pr(t,n,r,e){if(!_u(t))return t;n=Ae(n,t)?[n]:Sr(n);for(var u=-1,i=n.length,o=i-1,f=t;null!=f&&++u<i;){var c=Le(n[u]),a=r;if(u!=o){var l=f[c],a=e?e(l,c,f):F;a===F&&(a=_u(l)?l:we(n[u+1])?[]:{})}sn(f,c,a),f=f[c]}return t}function _r(t){return Be(Wu(t))}function vr(t,n,r){var e=-1,u=t.length;
for(0>n&&(n=-n>u?0:u+n),r=r>u?u:r,0>r&&(r+=u),u=n>r?0:r-n>>>0,n>>>=0,r=Zu(u);++e<u;)r[e]=t[e+n];return r}function gr(t,n){var r;return to(t,function(t,e,u){return r=n(t,e,u),!r}),!!r}function dr(t,n,r){var e=0,u=t?t.length:e;if(typeof n=="number"&&n===n&&2147483647>=u){for(;e<u;){var i=e+u>>>1,o=t[i];null!==o&&!bu(o)&&(r?o<=n:o<n)?e=i+1:u=i}return u}return yr(t,n,Mu,r)}function yr(t,n,r,e){n=r(n);for(var u=0,i=t?t.length:0,o=n!==n,f=null===n,c=bu(n),a=n===F;u<i;){var l=ki((u+i)/2),s=r(t[l]),h=s!==F,p=null===s,_=s===s,v=bu(s);
(o?e||_:a?_&&(e||h):f?_&&h&&(e||!p):c?_&&h&&!p&&(e||!v):p||v?0:e?s<=n:s<n)?u=l+1:i=l}return Wi(i,4294967294)}function br(t,n){for(var r=-1,e=t.length,u=0,i=[];++r<e;){var o=t[r],f=n?n(o):o;if(!r||!ou(f,c)){var c=f;i[u++]=0===o?0:o}}return i}function xr(t){return typeof t=="number"?t:bu(t)?P:+t}function jr(t){if(typeof t=="string")return t;if(bu(t))return Qi?Qi.call(t):"";var n=t+"";return"0"==n&&1/t==-N?"-0":n}function wr(t,n,r){var e=-1,u=c,i=t.length,o=true,f=[],l=f;if(r)o=false,u=a;else if(200<=i){if(u=n?null:fo(t))return M(u);
o=false,u=R,l=new qt}else l=n?[]:f;t:for(;++e<i;){var s=t[e],h=n?n(s):s,s=r||0!==s?s:0;if(o&&h===h){for(var p=l.length;p--;)if(l[p]===h)continue t;n&&l.push(h),f.push(s)}else u(l,h,r)||(l!==f&&l.push(h),f.push(s))}return f}function mr(t,n,r,e){for(var u=t.length,i=e?u:-1;(e?i--:++i<u)&&n(t[i],i,t););return r?vr(t,e?0:i,e?i+1:u):vr(t,e?i+1:0,e?u:i)}function Ar(t,n){var r=t;return r instanceof Dt&&(r=r.value()),h(n,function(t,n){return n.func.apply(n.thisArg,s([t],n.args))},r)}function kr(t,n,r){for(var e=-1,u=t.length;++e<u;)var i=i?s(wn(i,t[e],n,r),wn(t[e],i,n,r)):t[e];
return i&&i.length?wr(i,n,r):[]}function Er(t,n,r){for(var e=-1,u=t.length,i=n.length,o={};++e<u;)r(o,t[e],e<i?n[e]:F);return o}function Or(t){return au(t)?t:[]}function Sr(t){return tf(t)?t:vo(t)}function Ir(t,n,r){var e=t.length;return r=r===F?e:r,!n&&r>=e?t:vr(t,n,r)}function Rr(t,n){if(n)return t.slice();var r=t.length,r=pi?pi(r):new t.constructor(r);return t.copy(r),r}function zr(t){var n=new t.constructor(t.byteLength);return new hi(n).set(new hi(t)),n}function Wr(t,n){return new t.constructor(n?zr(t.buffer):t.buffer,t.byteOffset,t.length);
}function Br(t,n){if(t!==n){var r=t!==F,e=null===t,u=t===t,i=bu(t),o=n!==F,f=null===n,c=n===n,a=bu(n);if(!f&&!a&&!i&&t>n||i&&o&&c&&!f&&!a||e&&o&&c||!r&&c||!u)return 1;if(!e&&!i&&!a&&t<n||a&&r&&u&&!e&&!i||f&&r&&u||!o&&u||!c)return-1}return 0}function Lr(t,n,r,e){var u=-1,i=t.length,o=r.length,f=-1,c=n.length,a=zi(i-o,0),l=Zu(c+a);for(e=!e;++f<c;)l[f]=n[f];for(;++u<o;)(e||u<i)&&(l[r[u]]=t[u]);for(;a--;)l[f++]=t[u++];return l}function Ur(t,n,r,e){var u=-1,i=t.length,o=-1,f=r.length,c=-1,a=n.length,l=zi(i-f,0),s=Zu(l+a);
for(e=!e;++u<l;)s[u]=t[u];for(l=u;++c<a;)s[l+c]=n[c];for(;++o<f;)(e||u<i)&&(s[l+r[o]]=t[u++]);return s}function Cr(t,n){var r=-1,e=t.length;for(n||(n=Zu(e));++r<e;)n[r]=t[r];return n}function Mr(t,n,r,e){var u=!r;r||(r={});for(var i=-1,o=n.length;++i<o;){var f=n[i],c=e?e(r[f],t[f],f,r,t):F;c===F&&(c=t[f]),u?vn(r,f,c):sn(r,f,c)}return r}function Dr(t,n){return Mr(t,ao(t),n)}function Tr(t,n){return function(r,u){var i=tf(r)?e:pn,o=n?n():{};return i(r,t,pe(u,2),o)}}function $r(t){return lr(function(n,r){
var e=-1,u=r.length,i=1<u?r[u-1]:F,o=2<u?r[2]:F,i=3<t.length&&typeof i=="function"?(u--,i):F;for(o&&me(r[0],r[1],o)&&(i=3>u?F:i,u=1),n=Ju(n);++e<u;)(o=r[e])&&t(n,o,e,i);return n})}function Fr(t,n){return function(r,e){if(null==r)return r;if(!cu(r))return t(r,e);for(var u=r.length,i=n?u:-1,o=Ju(r);(n?i--:++i<u)&&false!==e(o[i],i,o););return r}}function Nr(t){return function(n,r,e){var u=-1,i=Ju(n);e=e(n);for(var o=e.length;o--;){var f=e[t?o:++u];if(false===r(i[f],f,i))break}return n}}function Pr(t,n,r){
function e(){return(this&&this!==Pt&&this instanceof e?i:t).apply(u?r:this,arguments)}var u=1&n,i=Vr(t);return e}function Zr(t){return function(n){n=Eu(n);var r=Wt.test(n)?$(n):F,e=r?r[0]:n.charAt(0);return n=r?Ir(r,1).join(""):n.slice(1),e[t]()+n}}function qr(t){return function(n){return h(Uu(Lu(n).replace(St,"")),t,"")}}function Vr(t){return function(){var n=arguments;switch(n.length){case 0:return new t;case 1:return new t(n[0]);case 2:return new t(n[0],n[1]);case 3:return new t(n[0],n[1],n[2]);
case 4:return new t(n[0],n[1],n[2],n[3]);case 5:return new t(n[0],n[1],n[2],n[3],n[4]);case 6:return new t(n[0],n[1],n[2],n[3],n[4],n[5]);case 7:return new t(n[0],n[1],n[2],n[3],n[4],n[5],n[6])}var r=Xi(t.prototype),n=t.apply(r,n);return _u(n)?n:r}}function Kr(t,n,e){function u(){for(var o=arguments.length,f=Zu(o),c=o,a=he(u);c--;)f[c]=arguments[c];return c=3>o&&f[0]!==a&&f[o-1]!==a?[]:C(f,a),o-=c.length,o<e?ue(t,n,Yr,u.placeholder,F,f,c,F,F,e-o):r(this&&this!==Pt&&this instanceof u?i:t,this,f)}var i=Vr(t);
return u}function Gr(t){return function(n,r,e){var u=Ju(n);if(!cu(n)){var i=pe(r,3);n=Iu(n),r=function(t){return i(u[t],t,u)}}return r=t(n,r,e),-1<r?u[i?n[r]:r]:F}}function Jr(t){return le(function(n){var r=n.length,e=r,u=Mt.prototype.thru;for(t&&n.reverse();e--;){var i=n[e];if(typeof i!="function")throw new Qu("Expected a function");if(u&&!o&&"wrapper"==se(i))var o=new Mt([],true)}for(e=o?e:r;++e<r;)var i=n[e],u=se(i),f="wrapper"==u?co(i):F,o=f&&ke(f[0])&&424==f[1]&&!f[4].length&&1==f[9]?o[se(f[0])].apply(o,f[3]):1==i.length&&ke(i)?o[u]():o.thru(i);
return function(){var t=arguments,e=t[0];if(o&&1==t.length&&tf(e)&&200<=e.length)return o.plant(e).value();for(var u=0,t=r?n[u].apply(this,t):e;++u<r;)t=n[u].call(this,t);return t}})}function Yr(t,n,r,e,u,i,o,f,c,a){function l(){for(var d=arguments.length,y=Zu(d),b=d;b--;)y[b]=arguments[b];if(_){var x,j=he(l),b=y.length;for(x=0;b--;)y[b]===j&&++x}if(e&&(y=Lr(y,e,u,_)),i&&(y=Ur(y,i,o,_)),d-=x,_&&d<a)return j=C(y,j),ue(t,n,Yr,l.placeholder,r,y,j,f,c,a-d);if(j=h?r:this,b=p?j[t]:t,d=y.length,f){x=y.length;
for(var w=Wi(f.length,x),m=Cr(y);w--;){var A=f[w];y[w]=we(A,x)?m[A]:F}}else v&&1<d&&y.reverse();return s&&c<d&&(y.length=c),this&&this!==Pt&&this instanceof l&&(b=g||Vr(b)),b.apply(j,y)}var s=128&n,h=1&n,p=2&n,_=24&n,v=512&n,g=p?F:Vr(t);return l}function Hr(t,n){return function(r,e){return Cn(r,t,n(e))}}function Qr(t,n){return function(r,e){var u;if(r===F&&e===F)return n;if(r!==F&&(u=r),e!==F){if(u===F)return e;typeof r=="string"||typeof e=="string"?(r=jr(r),e=jr(e)):(r=xr(r),e=xr(e)),u=t(r,e)}return u;
}}function Xr(t){return le(function(n){return n=l(n,S(pe())),lr(function(e){var u=this;return t(n,function(t){return r(t,u,e)})})})}function te(t,n){n=n===F?" ":jr(n);var r=n.length;return 2>r?r?ar(n,t):n:(r=ar(n,Ai(t/T(n))),Wt.test(n)?Ir($(r),0,t).join(""):r.slice(0,t))}function ne(t,n,e,u){function i(){for(var n=-1,c=arguments.length,a=-1,l=u.length,s=Zu(l+c),h=this&&this!==Pt&&this instanceof i?f:t;++a<l;)s[a]=u[a];for(;c--;)s[a++]=arguments[++n];return r(h,o?e:this,s)}var o=1&n,f=Vr(t);return i;
}function re(t){return function(n,r,e){e&&typeof e!="number"&&me(n,r,e)&&(r=e=F),n=ju(n),r===F?(r=n,n=0):r=ju(r),e=e===F?n<r?1:-1:ju(e);var u=-1;r=zi(Ai((r-n)/(e||1)),0);for(var i=Zu(r);r--;)i[t?r:++u]=n,n+=e;return i}}function ee(t){return function(n,r){return typeof n=="string"&&typeof r=="string"||(n=Au(n),r=Au(r)),t(n,r)}}function ue(t,n,r,e,u,i,o,f,c,a){var l=8&n,s=l?o:F;o=l?F:o;var h=l?i:F;return i=l?F:i,n=(n|(l?32:64))&~(l?64:32),4&n||(n&=-4),u=[t,n,u,h,s,i,o,f,c,a],r=r.apply(F,u),ke(t)&&ho(r,u),
r.placeholder=e,ze(r,t,n)}function ie(t){var n=Gu[t];return function(t,r){if(t=Au(t),r=Wi(wu(r),292)){var e=(Eu(t)+"e").split("e"),e=n(e[0]+"e"+(+e[1]+r)),e=(Eu(e)+"e").split("e");return+(e[0]+"e"+(+e[1]-r))}return n(t)}}function oe(t){return function(n){var r=Et(n);return"[object Map]"==r?L(n):"[object Set]"==r?D(n):O(n,t(n))}}function fe(t,n,r,e,u,i,o,f){var c=2&n;if(!c&&typeof t!="function")throw new Qu("Expected a function");var a=e?e.length:0;if(a||(n&=-97,e=u=F),o=o===F?o:zi(wu(o),0),f=f===F?f:wu(f),
a-=u?u.length:0,64&n){var l=e,s=u;e=u=F}var h=c?F:co(t);return i=[t,n,r,e,u,l,s,i,o,f],h&&(r=i[1],t=h[1],n=r|t,e=128==t&&8==r||128==t&&256==r&&i[7].length<=h[8]||384==t&&h[7].length<=h[8]&&8==r,131>n||e)&&(1&t&&(i[2]=h[2],n|=1&r?0:4),(r=h[3])&&(e=i[3],i[3]=e?Lr(e,r,h[4]):r,i[4]=e?C(i[3],"__lodash_placeholder__"):h[4]),(r=h[5])&&(e=i[5],i[5]=e?Ur(e,r,h[6]):r,i[6]=e?C(i[5],"__lodash_placeholder__"):h[6]),(r=h[7])&&(i[7]=r),128&t&&(i[8]=null==i[8]?h[8]:Wi(i[8],h[8])),null==i[9]&&(i[9]=h[9]),i[0]=h[0],
i[1]=n),t=i[0],n=i[1],r=i[2],e=i[3],u=i[4],f=i[9]=null==i[9]?c?0:t.length:zi(i[9]-a,0),!f&&24&n&&(n&=-25),ze((h?uo:ho)(n&&1!=n?8==n||16==n?Kr(t,n,f):32!=n&&33!=n||u.length?Yr.apply(F,i):ne(t,n,r,e):Pr(t,n,r),i),t,n)}function ce(t,n,r,e,u,i){var o=2&u,f=t.length,c=n.length;if(f!=c&&!(o&&c>f))return false;if((c=i.get(t))&&i.get(n))return c==n;var c=-1,a=true,l=1&u?new qt:F;for(i.set(t,n),i.set(n,t);++c<f;){var s=t[c],h=n[c];if(e)var p=o?e(h,s,c,n,t,i):e(s,h,c,t,n,i);if(p!==F){if(p)continue;a=false;break}if(l){
if(!_(n,function(t,n){if(!R(l,n)&&(s===t||r(s,t,e,u,i)))return l.push(n)})){a=false;break}}else if(s!==h&&!r(s,h,e,u,i)){a=false;break}}return i.delete(t),i.delete(n),a}function ae(t,n,r,e,u,i,o){switch(r){case"[object DataView]":if(t.byteLength!=n.byteLength||t.byteOffset!=n.byteOffset)break;t=t.buffer,n=n.buffer;case"[object ArrayBuffer]":if(t.byteLength!=n.byteLength||!e(new hi(t),new hi(n)))break;return true;case"[object Boolean]":case"[object Date]":case"[object Number]":return ou(+t,+n);case"[object Error]":
return t.name==n.name&&t.message==n.message;case"[object RegExp]":case"[object String]":return t==n+"";case"[object Map]":var f=L;case"[object Set]":if(f||(f=M),t.size!=n.size&&!(2&i))break;return(r=o.get(t))?r==n:(i|=1,o.set(t,n),n=ce(f(t),f(n),e,u,i,o),o.delete(t),n);case"[object Symbol]":if(Hi)return Hi.call(t)==Hi.call(n)}return false}function le(t){return _o(Ie(t,F,$e),t+"")}function se(t){for(var n=t.name+"",r=Zi[n],e=ui.call(Zi,n)?r.length:0;e--;){var u=r[e],i=u.func;if(null==i||i==t)return u.name;
}return n}function he(t){return(ui.call(Ot,"placeholder")?Ot:t).placeholder}function pe(){var t=Ot.iteratee||Du,t=t===Du?Kn:t;return arguments.length?t(arguments[0],arguments[1]):t}function _e(t,n){var r=t.__data__,e=typeof n;return("string"==e||"number"==e||"symbol"==e||"boolean"==e?"__proto__"!==n:null===n)?r[typeof n=="string"?"string":"hash"]:r.map}function ve(t){for(var n=Iu(t),r=n.length;r--;){var e=n[r],u=t[e];n[r]=[e,u,u===u&&!_u(u)]}return n}function ge(t,n){var r=null==t?F:t[n];return Pn(r)?r:F;
}function de(t,n,r){n=Ae(n,t)?[n]:Sr(n);for(var e=-1,u=n.length,i=false;++e<u;){var o=Le(n[e]);if(!(i=null!=t&&r(t,o)))break;t=t[o]}return i||++e!=u?i:(u=t?t.length:0,!!u&&pu(u)&&we(o,u)&&(tf(t)||fu(t)))}function ye(t){var n=t.length,r=t.constructor(n);return n&&"string"==typeof t[0]&&ui.call(t,"index")&&(r.index=t.index,r.input=t.input),r}function be(t){return typeof t.constructor!="function"||Ee(t)?{}:Xi(_i(t))}function xe(r,e,u,i){var o=r.constructor;switch(e){case"[object ArrayBuffer]":return zr(r);
case"[object Boolean]":case"[object Date]":return new o(+r);case"[object DataView]":return e=i?zr(r.buffer):r.buffer,new r.constructor(e,r.byteOffset,r.byteLength);case"[object Float32Array]":case"[object Float64Array]":case"[object Int8Array]":case"[object Int16Array]":case"[object Int32Array]":case"[object Uint8Array]":case"[object Uint8ClampedArray]":case"[object Uint16Array]":case"[object Uint32Array]":return Wr(r,i);case"[object Map]":return e=i?u(L(r),true):L(r),h(e,t,new r.constructor);case"[object Number]":
case"[object String]":return new o(r);case"[object RegExp]":return e=new r.constructor(r.source,gt.exec(r)),e.lastIndex=r.lastIndex,e;case"[object Set]":return e=i?u(M(r),true):M(r),h(e,n,new r.constructor);case"[object Symbol]":return Hi?Ju(Hi.call(r)):{}}}function je(t){return tf(t)||fu(t)||!!(bi&&t&&t[bi])}function we(t,n){return n=null==n?9007199254740991:n,!!n&&(typeof t=="number"||jt.test(t))&&-1<t&&0==t%1&&t<n}function me(t,n,r){if(!_u(r))return false;var e=typeof n;return!!("number"==e?cu(r)&&we(n,r.length):"string"==e&&n in r)&&ou(r[n],t);
}function Ae(t,n){if(tf(t))return false;var r=typeof t;return!("number"!=r&&"symbol"!=r&&"boolean"!=r&&null!=t&&!bu(t))||(rt.test(t)||!nt.test(t)||null!=n&&t in Ju(n))}function ke(t){var n=se(t),r=Ot[n];return typeof r=="function"&&n in Dt.prototype&&(t===r||(n=co(r),!!n&&t===n[0]))}function Ee(t){var n=t&&t.constructor;return t===(typeof n=="function"&&n.prototype||ti)}function Oe(t,n){return function(r){return null!=r&&(r[t]===n&&(n!==F||t in Ju(r)))}}function Se(t,n,r,e,u,i){return _u(t)&&_u(n)&&(i.set(n,t),
tr(t,n,F,Se,i),i.delete(n)),t}function Ie(t,n,e){return n=zi(n===F?t.length-1:n,0),function(){for(var u=arguments,i=-1,o=zi(u.length-n,0),f=Zu(o);++i<o;)f[i]=u[n+i];for(i=-1,o=Zu(n+1);++i<n;)o[i]=u[i];return o[n]=e(f),r(t,this,o)}}function Re(t,n){return 1==n.length?t:Rn(t,vr(n,0,-1))}function ze(t,n,r){var e=n+"";n=_o;var u,i=Ce;return u=(u=e.match(st))?u[1].split(ht):[],r=i(u,r),(i=r.length)&&(u=i-1,r[u]=(1<i?"& ":"")+r[u],r=r.join(2<i?", ":" "),e=e.replace(lt,"{\n/* [wrapped with "+r+"] */\n")),
n(t,e)}function We(t){var n=0,r=0;return function(){var e=Bi(),u=16-(e-r);if(r=e,0<u){if(500<=++n)return arguments[0]}else n=0;return t.apply(F,arguments)}}function Be(t,n){var r=-1,e=t.length,u=e-1;for(n=n===F?e:dn(n,0,e);++r<n;){var e=cr(r,u),i=t[e];t[e]=t[r],t[r]=i}return t.length=n,t}function Le(t){if(typeof t=="string"||bu(t))return t;var n=t+"";return"0"==n&&1/t==-N?"-0":n}function Ue(t){if(null!=t){try{return ei.call(t)}catch(t){}return t+""}return""}function Ce(t,n){return u(Z,function(r){
var e="_."+r[0];n&r[1]&&!c(t,e)&&t.push(e)}),t.sort()}function Me(t){if(t instanceof Dt)return t.clone();var n=new Mt(t.__wrapped__,t.__chain__);return n.__actions__=Cr(t.__actions__),n.__index__=t.__index__,n.__values__=t.__values__,n}function De(t,n,r){var e=t?t.length:0;return e?(r=null==r?0:wu(r),0>r&&(r=zi(e+r,0)),g(t,pe(n,3),r)):-1}function Te(t,n,r){var e=t?t.length:0;if(!e)return-1;var u=e-1;return r!==F&&(u=wu(r),u=0>r?zi(e+u,0):Wi(u,e-1)),g(t,pe(n,3),u,true)}function $e(t){return t&&t.length?En(t,1):[];
}function Fe(t){return t&&t.length?t[0]:F}function Ne(t){var n=t?t.length:0;return n?t[n-1]:F}function Pe(t,n){return t&&t.length&&n&&n.length?or(t,n):t}function Ze(t){return t?Ci.call(t):t}function qe(t){if(!t||!t.length)return[];var n=0;return t=f(t,function(t){if(au(t))return n=zi(t.length,n),true}),E(n,function(n){return l(t,j(n))})}function Ve(t,n){if(!t||!t.length)return[];var e=qe(t);return null==n?e:l(e,function(t){return r(n,F,t)})}function Ke(t){return t=Ot(t),t.__chain__=true,t}function Ge(t,n){
return n(t)}function Je(){return this}function Ye(t,n){return(tf(t)?u:to)(t,pe(n,3))}function He(t,n){return(tf(t)?i:no)(t,pe(n,3))}function Qe(t,n){return(tf(t)?l:Hn)(t,pe(n,3))}function Xe(t,n,r){return n=r?F:n,n=t&&null==n?t.length:n,fe(t,128,F,F,F,F,n)}function tu(t,n){var r;if(typeof n!="function")throw new Qu("Expected a function");return t=wu(t),function(){return 0<--t&&(r=n.apply(this,arguments)),1>=t&&(n=F),r}}function nu(t,n,r){return n=r?F:n,t=fe(t,8,F,F,F,F,F,n),t.placeholder=nu.placeholder,
t}function ru(t,n,r){return n=r?F:n,t=fe(t,16,F,F,F,F,F,n),t.placeholder=ru.placeholder,t}function eu(t,n,r){function e(n){var r=c,e=a;return c=a=F,_=n,s=t.apply(e,r)}function u(t){var r=t-p;return t-=_,p===F||r>=n||0>r||g&&t>=l}function i(){var t=Po();if(u(t))return o(t);var r,e=po;r=t-_,t=n-(t-p),r=g?Wi(t,l-r):t,h=e(i,r)}function o(t){return h=F,d&&c?e(t):(c=a=F,s)}function f(){var t=Po(),r=u(t);if(c=arguments,a=this,p=t,r){if(h===F)return _=t=p,h=po(i,n),v?e(t):s;if(g)return h=po(i,n),e(p)}return h===F&&(h=po(i,n)),
s}var c,a,l,s,h,p,_=0,v=false,g=false,d=true;if(typeof t!="function")throw new Qu("Expected a function");return n=Au(n)||0,_u(r)&&(v=!!r.leading,l=(g="maxWait"in r)?zi(Au(r.maxWait)||0,n):l,d="trailing"in r?!!r.trailing:d),f.cancel=function(){h!==F&&oo(h),_=0,c=p=a=h=F},f.flush=function(){return h===F?s:o(Po())},f}function uu(t,n){function r(){var e=arguments,u=n?n.apply(this,e):e[0],i=r.cache;return i.has(u)?i.get(u):(e=t.apply(this,e),r.cache=i.set(u,e)||i,e)}if(typeof t!="function"||n&&typeof n!="function")throw new Qu("Expected a function");
return r.cache=new(uu.Cache||Zt),r}function iu(t){if(typeof t!="function")throw new Qu("Expected a function");return function(){var n=arguments;switch(n.length){case 0:return!t.call(this);case 1:return!t.call(this,n[0]);case 2:return!t.call(this,n[0],n[1]);case 3:return!t.call(this,n[0],n[1],n[2])}return!t.apply(this,n)}}function ou(t,n){return t===n||t!==t&&n!==n}function fu(t){return au(t)&&ui.call(t,"callee")&&(!di.call(t,"callee")||"[object Arguments]"==fi.call(t))}function cu(t){return null!=t&&pu(t.length)&&!su(t);
}function au(t){return vu(t)&&cu(t)}function lu(t){return!!vu(t)&&("[object Error]"==fi.call(t)||typeof t.message=="string"&&typeof t.name=="string")}function su(t){return t=_u(t)?fi.call(t):"","[object Function]"==t||"[object GeneratorFunction]"==t||"[object Proxy]"==t}function hu(t){return typeof t=="number"&&t==wu(t)}function pu(t){return typeof t=="number"&&-1<t&&0==t%1&&9007199254740991>=t}function _u(t){var n=typeof t;return null!=t&&("object"==n||"function"==n)}function vu(t){return null!=t&&typeof t=="object";
}function gu(t){return typeof t=="number"||vu(t)&&"[object Number]"==fi.call(t)}function du(t){return!(!vu(t)||"[object Object]"!=fi.call(t))&&(t=_i(t),null===t||(t=ui.call(t,"constructor")&&t.constructor,typeof t=="function"&&t instanceof t&&ei.call(t)==oi))}function yu(t){return typeof t=="string"||!tf(t)&&vu(t)&&"[object String]"==fi.call(t)}function bu(t){return typeof t=="symbol"||vu(t)&&"[object Symbol]"==fi.call(t)}function xu(t){if(!t)return[];if(cu(t))return yu(t)?$(t):Cr(t);if(vi&&t[vi]){t=t[vi]();
for(var n,r=[];!(n=t.next()).done;)r.push(n.value);return r}return n=Et(t),("[object Map]"==n?L:"[object Set]"==n?M:Wu)(t)}function ju(t){return t?(t=Au(t),t===N||t===-N?1.7976931348623157e308*(0>t?-1:1):t===t?t:0):0===t?t:0}function wu(t){t=ju(t);var n=t%1;return t===t?n?t-n:t:0}function mu(t){return t?dn(wu(t),0,4294967295):0}function Au(t){if(typeof t=="number")return t;if(bu(t))return P;if(_u(t)&&(t=typeof t.valueOf=="function"?t.valueOf():t,t=_u(t)?t+"":t),typeof t!="string")return 0===t?t:+t;
t=t.replace(ft,"");var n=yt.test(t);return n||xt.test(t)?$t(t.slice(2),n?2:8):dt.test(t)?P:+t}function ku(t){return Mr(t,Ru(t))}function Eu(t){return null==t?"":jr(t)}function Ou(t,n,r){return t=null==t?F:Rn(t,n),t===F?r:t}function Su(t,n){return null!=t&&de(t,n,Ln)}function Iu(t){return cu(t)?tn(t):Gn(t)}function Ru(t){return cu(t)?tn(t,true):Jn(t)}function zu(t,n){return null==t?{}:ur(t,zn(t,Ru,lo),pe(n))}function Wu(t){return t?I(t,Iu(t)):[]}function Bu(t){return Lf(Eu(t).toLowerCase())}function Lu(t){
return(t=Eu(t))&&t.replace(wt,nn).replace(It,"")}function Uu(t,n,r){return t=Eu(t),n=r?F:n,n===F?Bt.test(t)?t.match(zt)||[]:t.match(pt)||[]:t.match(n)||[]}function Cu(t){return function(){return t}}function Mu(t){return t}function Du(t){return Kn(typeof t=="function"?t:yn(t,true))}function Tu(t,n,r){var e=Iu(n),i=In(n,e);null!=r||_u(n)&&(i.length||!e.length)||(r=n,n=t,t=this,i=In(n,Iu(n)));var o=!(_u(r)&&"chain"in r&&!r.chain),f=su(t);return u(i,function(r){var e=n[r];t[r]=e,f&&(t.prototype[r]=function(){
var n=this.__chain__;if(o||n){var r=t(this.__wrapped__);return(r.__actions__=Cr(this.__actions__)).push({func:e,args:arguments,thisArg:t}),r.__chain__=n,r}return e.apply(t,s([this.value()],arguments))})}),t}function $u(){}function Fu(t){return Ae(t)?j(Le(t)):ir(t)}function Nu(){return[]}function Pu(){return false}kt=kt?un.defaults(Pt.Object(),kt,un.pick(Pt,Lt)):Pt;var Zu=kt.Array,qu=kt.Date,Vu=kt.Error,Ku=kt.Function,Gu=kt.Math,Ju=kt.Object,Yu=kt.RegExp,Hu=kt.String,Qu=kt.TypeError,Xu=Zu.prototype,ti=Ju.prototype,ni=kt["__core-js_shared__"],ri=function(){
var t=/[^.]+$/.exec(ni&&ni.keys&&ni.keys.IE_PROTO||"");return t?"Symbol(src)_1."+t:""}(),ei=Ku.prototype.toString,ui=ti.hasOwnProperty,ii=0,oi=ei.call(Ju),fi=ti.toString,ci=Pt._,ai=Yu("^"+ei.call(ui).replace(it,"\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g,"$1.*?")+"$"),li=Vt?kt.Buffer:F,si=kt.Symbol,hi=kt.Uint8Array,pi=li?li.f:F,_i=U(Ju.getPrototypeOf,Ju),vi=si?si.iterator:F,gi=Ju.create,di=ti.propertyIsEnumerable,yi=Xu.splice,bi=si?si.isConcatSpreadable:F,xi=function(){
try{var t=ge(Ju,"defineProperty");return t({},"",{}),t}catch(t){}}(),ji=kt.clearTimeout!==Pt.clearTimeout&&kt.clearTimeout,wi=qu&&qu.now!==Pt.Date.now&&qu.now,mi=kt.setTimeout!==Pt.setTimeout&&kt.setTimeout,Ai=Gu.ceil,ki=Gu.floor,Ei=Ju.getOwnPropertySymbols,Oi=li?li.isBuffer:F,Si=kt.isFinite,Ii=Xu.join,Ri=U(Ju.keys,Ju),zi=Gu.max,Wi=Gu.min,Bi=qu.now,Li=kt.parseInt,Ui=Gu.random,Ci=Xu.reverse,Mi=ge(kt,"DataView"),Di=ge(kt,"Map"),Ti=ge(kt,"Promise"),$i=ge(kt,"Set"),Fi=ge(kt,"WeakMap"),Ni=ge(Ju,"create"),Pi=Fi&&new Fi,Zi={},qi=Ue(Mi),Vi=Ue(Di),Ki=Ue(Ti),Gi=Ue($i),Ji=Ue(Fi),Yi=si?si.prototype:F,Hi=Yi?Yi.valueOf:F,Qi=Yi?Yi.toString:F,Xi=function(){
function t(){}return function(n){return _u(n)?gi?gi(n):(t.prototype=n,n=new t,t.prototype=F,n):{}}}();Ot.templateSettings={escape:Q,evaluate:X,interpolate:tt,variable:"",imports:{_:Ot}},Ot.prototype=Rt.prototype,Ot.prototype.constructor=Ot,Mt.prototype=Xi(Rt.prototype),Mt.prototype.constructor=Mt,Dt.prototype=Xi(Rt.prototype),Dt.prototype.constructor=Dt,Ft.prototype.clear=function(){this.__data__=Ni?Ni(null):{},this.size=0},Ft.prototype.delete=function(t){return t=this.has(t)&&delete this.__data__[t],
this.size-=t?1:0,t},Ft.prototype.get=function(t){var n=this.__data__;return Ni?(t=n[t],"__lodash_hash_undefined__"===t?F:t):ui.call(n,t)?n[t]:F},Ft.prototype.has=function(t){var n=this.__data__;return Ni?n[t]!==F:ui.call(n,t)},Ft.prototype.set=function(t,n){var r=this.__data__;return this.size+=this.has(t)?0:1,r[t]=Ni&&n===F?"__lodash_hash_undefined__":n,this},Nt.prototype.clear=function(){this.__data__=[],this.size=0},Nt.prototype.delete=function(t){var n=this.__data__;return t=hn(n,t),!(0>t)&&(t==n.length-1?n.pop():yi.call(n,t,1),
--this.size,true)},Nt.prototype.get=function(t){var n=this.__data__;return t=hn(n,t),0>t?F:n[t][1]},Nt.prototype.has=function(t){return-1<hn(this.__data__,t)},Nt.prototype.set=function(t,n){var r=this.__data__,e=hn(r,t);return 0>e?(++this.size,r.push([t,n])):r[e][1]=n,this},Zt.prototype.clear=function(){this.size=0,this.__data__={hash:new Ft,map:new(Di||Nt),string:new Ft}},Zt.prototype.delete=function(t){return t=_e(this,t).delete(t),this.size-=t?1:0,t},Zt.prototype.get=function(t){return _e(this,t).get(t);
},Zt.prototype.has=function(t){return _e(this,t).has(t)},Zt.prototype.set=function(t,n){var r=_e(this,t),e=r.size;return r.set(t,n),this.size+=r.size==e?0:1,this},qt.prototype.add=qt.prototype.push=function(t){return this.__data__.set(t,"__lodash_hash_undefined__"),this},qt.prototype.has=function(t){return this.__data__.has(t)},Kt.prototype.clear=function(){this.__data__=new Nt,this.size=0},Kt.prototype.delete=function(t){var n=this.__data__;return t=n.delete(t),this.size=n.size,t},Kt.prototype.get=function(t){
return this.__data__.get(t)},Kt.prototype.has=function(t){return this.__data__.has(t)},Kt.prototype.set=function(t,n){var r=this.__data__;if(r instanceof Nt){var e=r.__data__;if(!Di||199>e.length)return e.push([t,n]),this.size=++r.size,this;r=this.__data__=new Zt(e)}return r.set(t,n),this.size=r.size,this};var to=Fr(On),no=Fr(Sn,true),ro=Nr(),eo=Nr(true),uo=Pi?function(t,n){return Pi.set(t,n),t}:Mu,io=xi?function(t,n){return xi(t,"toString",{configurable:true,enumerable:false,value:Cu(n),writable:true})}:Mu,oo=ji||function(t){
return Pt.clearTimeout(t)},fo=$i&&1/M(new $i([,-0]))[1]==N?function(t){return new $i(t)}:$u,co=Pi?function(t){return Pi.get(t)}:$u,ao=Ei?U(Ei,Ju):Nu,lo=Ei?function(t){for(var n=[];t;)s(n,ao(t)),t=_i(t);return n}:Nu;(Mi&&"[object DataView]"!=Et(new Mi(new ArrayBuffer(1)))||Di&&"[object Map]"!=Et(new Di)||Ti&&"[object Promise]"!=Et(Ti.resolve())||$i&&"[object Set]"!=Et(new $i)||Fi&&"[object WeakMap]"!=Et(new Fi))&&(Et=function(t){var n=fi.call(t);if(t=(t="[object Object]"==n?t.constructor:F)?Ue(t):F)switch(t){
case qi:return"[object DataView]";case Vi:return"[object Map]";case Ki:return"[object Promise]";case Gi:return"[object Set]";case Ji:return"[object WeakMap]"}return n});var so=ni?su:Pu,ho=We(uo),po=mi||function(t,n){return Pt.setTimeout(t,n)},_o=We(io),vo=function(t){t=uu(t,function(t){return 500===n.size&&n.clear(),t});var n=t.cache;return t}(function(t){t=Eu(t);var n=[];return et.test(t)&&n.push(""),t.replace(ut,function(t,r,e,u){n.push(e?u.replace(_t,"$1"):r||t)}),n}),go=lr(function(t,n){return au(t)?wn(t,En(n,1,au,true)):[];
}),yo=lr(function(t,n){var r=Ne(n);return au(r)&&(r=F),au(t)?wn(t,En(n,1,au,true),pe(r,2)):[]}),bo=lr(function(t,n){var r=Ne(n);return au(r)&&(r=F),au(t)?wn(t,En(n,1,au,true),F,r):[]}),xo=lr(function(t){var n=l(t,Or);return n.length&&n[0]===t[0]?Un(n):[]}),jo=lr(function(t){var n=Ne(t),r=l(t,Or);return n===Ne(r)?n=F:r.pop(),r.length&&r[0]===t[0]?Un(r,pe(n,2)):[]}),wo=lr(function(t){var n=Ne(t),r=l(t,Or);return n===Ne(r)?n=F:r.pop(),r.length&&r[0]===t[0]?Un(r,F,n):[]}),mo=lr(Pe),Ao=le(function(t,n){var r=t?t.length:0,e=gn(t,n);
return fr(t,l(n,function(t){return we(t,r)?+t:t}).sort(Br)),e}),ko=lr(function(t){return wr(En(t,1,au,true))}),Eo=lr(function(t){var n=Ne(t);return au(n)&&(n=F),wr(En(t,1,au,true),pe(n,2))}),Oo=lr(function(t){var n=Ne(t);return au(n)&&(n=F),wr(En(t,1,au,true),F,n)}),So=lr(function(t,n){return au(t)?wn(t,n):[]}),Io=lr(function(t){return kr(f(t,au))}),Ro=lr(function(t){var n=Ne(t);return au(n)&&(n=F),kr(f(t,au),pe(n,2))}),zo=lr(function(t){var n=Ne(t);return au(n)&&(n=F),kr(f(t,au),F,n)}),Wo=lr(qe),Bo=lr(function(t){
var n=t.length,n=1<n?t[n-1]:F,n=typeof n=="function"?(t.pop(),n):F;return Ve(t,n)}),Lo=le(function(t){function n(n){return gn(n,t)}var r=t.length,e=r?t[0]:0,u=this.__wrapped__;return!(1<r||this.__actions__.length)&&u instanceof Dt&&we(e)?(u=u.slice(e,+e+(r?1:0)),u.__actions__.push({func:Ge,args:[n],thisArg:F}),new Mt(u,this.__chain__).thru(function(t){return r&&!t.length&&t.push(F),t})):this.thru(n)}),Uo=Tr(function(t,n,r){ui.call(t,r)?++t[r]:vn(t,r,1)}),Co=Gr(De),Mo=Gr(Te),Do=Tr(function(t,n,r){
ui.call(t,r)?t[r].push(n):vn(t,r,[n])}),To=lr(function(t,n,e){var u=-1,i=typeof n=="function",o=Ae(n),f=cu(t)?Zu(t.length):[];return to(t,function(t){var c=i?n:o&&null!=t?t[n]:F;f[++u]=c?r(c,t,e):Mn(t,n,e)}),f}),$o=Tr(function(t,n,r){vn(t,r,n)}),Fo=Tr(function(t,n,r){t[r?0:1].push(n)},function(){return[[],[]]}),No=lr(function(t,n){if(null==t)return[];var r=n.length;return 1<r&&me(t,n[0],n[1])?n=[]:2<r&&me(n[0],n[1],n[2])&&(n=[n[0]]),rr(t,En(n,1),[])}),Po=wi||function(){return Pt.Date.now()},Zo=lr(function(t,n,r){
var e=1;if(r.length)var u=C(r,he(Zo)),e=32|e;return fe(t,e,n,r,u)}),qo=lr(function(t,n,r){var e=3;if(r.length)var u=C(r,he(qo)),e=32|e;return fe(n,e,t,r,u)}),Vo=lr(function(t,n){return jn(t,1,n)}),Ko=lr(function(t,n,r){return jn(t,Au(n)||0,r)});uu.Cache=Zt;var Go=lr(function(t,n){n=1==n.length&&tf(n[0])?l(n[0],S(pe())):l(En(n,1),S(pe()));var e=n.length;return lr(function(u){for(var i=-1,o=Wi(u.length,e);++i<o;)u[i]=n[i].call(this,u[i]);return r(t,this,u)})}),Jo=lr(function(t,n){return fe(t,32,F,n,C(n,he(Jo)));
}),Yo=lr(function(t,n){return fe(t,64,F,n,C(n,he(Yo)))}),Ho=le(function(t,n){return fe(t,256,F,F,F,n)}),Qo=ee(Wn),Xo=ee(function(t,n){return t>=n}),tf=Zu.isArray,nf=Gt?S(Gt):Dn,rf=Oi||Pu,ef=Jt?S(Jt):Tn,uf=Yt?S(Yt):Fn,of=Ht?S(Ht):Zn,ff=Qt?S(Qt):qn,cf=Xt?S(Xt):Vn,af=ee(Yn),lf=ee(function(t,n){return t<=n}),sf=$r(function(t,n){if(Ee(n)||cu(n))Mr(n,Iu(n),t);else for(var r in n)ui.call(n,r)&&sn(t,r,n[r])}),hf=$r(function(t,n){Mr(n,Ru(n),t)}),pf=$r(function(t,n,r,e){Mr(n,Ru(n),t,e)}),_f=$r(function(t,n,r,e){
Mr(n,Iu(n),t,e)}),vf=le(gn),gf=lr(function(t){return t.push(F,an),r(pf,F,t)}),df=lr(function(t){return t.push(F,Se),r(wf,F,t)}),yf=Hr(function(t,n,r){t[n]=r},Cu(Mu)),bf=Hr(function(t,n,r){ui.call(t,n)?t[n].push(r):t[n]=[r]},pe),xf=lr(Mn),jf=$r(function(t,n,r){tr(t,n,r)}),wf=$r(function(t,n,r,e){tr(t,n,r,e)}),mf=le(function(t,n){return null==t?{}:(n=l(n,Le),er(t,wn(zn(t,Ru,lo),n)))}),Af=le(function(t,n){return null==t?{}:er(t,l(n,Le))}),kf=oe(Iu),Ef=oe(Ru),Of=qr(function(t,n,r){return n=n.toLowerCase(),
t+(r?Bu(n):n)}),Sf=qr(function(t,n,r){return t+(r?"-":"")+n.toLowerCase()}),If=qr(function(t,n,r){return t+(r?" ":"")+n.toLowerCase()}),Rf=Zr("toLowerCase"),zf=qr(function(t,n,r){return t+(r?"_":"")+n.toLowerCase()}),Wf=qr(function(t,n,r){return t+(r?" ":"")+Lf(n)}),Bf=qr(function(t,n,r){return t+(r?" ":"")+n.toUpperCase()}),Lf=Zr("toUpperCase"),Uf=lr(function(t,n){try{return r(t,F,n)}catch(t){return lu(t)?t:new Vu(t)}}),Cf=le(function(t,n){return u(n,function(n){n=Le(n),vn(t,n,Zo(t[n],t))}),t}),Mf=Jr(),Df=Jr(true),Tf=lr(function(t,n){
return function(r){return Mn(r,t,n)}}),$f=lr(function(t,n){return function(r){return Mn(t,r,n)}}),Ff=Xr(l),Nf=Xr(o),Pf=Xr(_),Zf=re(),qf=re(true),Vf=Qr(function(t,n){return t+n},0),Kf=ie("ceil"),Gf=Qr(function(t,n){return t/n},1),Jf=ie("floor"),Yf=Qr(function(t,n){return t*n},1),Hf=ie("round"),Qf=Qr(function(t,n){return t-n},0);return Ot.after=function(t,n){if(typeof n!="function")throw new Qu("Expected a function");return t=wu(t),function(){if(1>--t)return n.apply(this,arguments)}},Ot.ary=Xe,Ot.assign=sf,
Ot.assignIn=hf,Ot.assignInWith=pf,Ot.assignWith=_f,Ot.at=vf,Ot.before=tu,Ot.bind=Zo,Ot.bindAll=Cf,Ot.bindKey=qo,Ot.castArray=function(){if(!arguments.length)return[];var t=arguments[0];return tf(t)?t:[t]},Ot.chain=Ke,Ot.chunk=function(t,n,r){if(n=(r?me(t,n,r):n===F)?1:zi(wu(n),0),r=t?t.length:0,!r||1>n)return[];for(var e=0,u=0,i=Zu(Ai(r/n));e<r;)i[u++]=vr(t,e,e+=n);return i},Ot.compact=function(t){for(var n=-1,r=t?t.length:0,e=0,u=[];++n<r;){var i=t[n];i&&(u[e++]=i)}return u},Ot.concat=function(){
var t=arguments.length;if(!t)return[];for(var n=Zu(t-1),r=arguments[0];t--;)n[t-1]=arguments[t];return s(tf(r)?Cr(r):[r],En(n,1))},Ot.cond=function(t){var n=t?t.length:0,e=pe();return t=n?l(t,function(t){if("function"!=typeof t[1])throw new Qu("Expected a function");return[e(t[0]),t[1]]}):[],lr(function(e){for(var u=-1;++u<n;){var i=t[u];if(r(i[0],this,e))return r(i[1],this,e)}})},Ot.conforms=function(t){return bn(yn(t,true))},Ot.constant=Cu,Ot.countBy=Uo,Ot.create=function(t,n){var r=Xi(t);return n?_n(r,n):r;
},Ot.curry=nu,Ot.curryRight=ru,Ot.debounce=eu,Ot.defaults=gf,Ot.defaultsDeep=df,Ot.defer=Vo,Ot.delay=Ko,Ot.difference=go,Ot.differenceBy=yo,Ot.differenceWith=bo,Ot.drop=function(t,n,r){var e=t?t.length:0;return e?(n=r||n===F?1:wu(n),vr(t,0>n?0:n,e)):[]},Ot.dropRight=function(t,n,r){var e=t?t.length:0;return e?(n=r||n===F?1:wu(n),n=e-n,vr(t,0,0>n?0:n)):[]},Ot.dropRightWhile=function(t,n){return t&&t.length?mr(t,pe(n,3),true,true):[]},Ot.dropWhile=function(t,n){return t&&t.length?mr(t,pe(n,3),true):[]},Ot.fill=function(t,n,r,e){
var u=t?t.length:0;if(!u)return[];for(r&&typeof r!="number"&&me(t,n,r)&&(r=0,e=u),u=t.length,r=wu(r),0>r&&(r=-r>u?0:u+r),e=e===F||e>u?u:wu(e),0>e&&(e+=u),e=r>e?0:mu(e);r<e;)t[r++]=n;return t},Ot.filter=function(t,n){return(tf(t)?f:kn)(t,pe(n,3))},Ot.flatMap=function(t,n){return En(Qe(t,n),1)},Ot.flatMapDeep=function(t,n){return En(Qe(t,n),N)},Ot.flatMapDepth=function(t,n,r){return r=r===F?1:wu(r),En(Qe(t,n),r)},Ot.flatten=$e,Ot.flattenDeep=function(t){return t&&t.length?En(t,N):[]},Ot.flattenDepth=function(t,n){
return t&&t.length?(n=n===F?1:wu(n),En(t,n)):[]},Ot.flip=function(t){return fe(t,512)},Ot.flow=Mf,Ot.flowRight=Df,Ot.fromPairs=function(t){for(var n=-1,r=t?t.length:0,e={};++n<r;){var u=t[n];e[u[0]]=u[1]}return e},Ot.functions=function(t){return null==t?[]:In(t,Iu(t))},Ot.functionsIn=function(t){return null==t?[]:In(t,Ru(t))},Ot.groupBy=Do,Ot.initial=function(t){return t&&t.length?vr(t,0,-1):[]},Ot.intersection=xo,Ot.intersectionBy=jo,Ot.intersectionWith=wo,Ot.invert=yf,Ot.invertBy=bf,Ot.invokeMap=To,
Ot.iteratee=Du,Ot.keyBy=$o,Ot.keys=Iu,Ot.keysIn=Ru,Ot.map=Qe,Ot.mapKeys=function(t,n){var r={};return n=pe(n,3),On(t,function(t,e,u){vn(r,n(t,e,u),t)}),r},Ot.mapValues=function(t,n){var r={};return n=pe(n,3),On(t,function(t,e,u){vn(r,e,n(t,e,u))}),r},Ot.matches=function(t){return Qn(yn(t,true))},Ot.matchesProperty=function(t,n){return Xn(t,yn(n,true))},Ot.memoize=uu,Ot.merge=jf,Ot.mergeWith=wf,Ot.method=Tf,Ot.methodOf=$f,Ot.mixin=Tu,Ot.negate=iu,Ot.nthArg=function(t){return t=wu(t),lr(function(n){return nr(n,t);
})},Ot.omit=mf,Ot.omitBy=function(t,n){return zu(t,iu(pe(n)))},Ot.once=function(t){return tu(2,t)},Ot.orderBy=function(t,n,r,e){return null==t?[]:(tf(n)||(n=null==n?[]:[n]),r=e?F:r,tf(r)||(r=null==r?[]:[r]),rr(t,n,r))},Ot.over=Ff,Ot.overArgs=Go,Ot.overEvery=Nf,Ot.overSome=Pf,Ot.partial=Jo,Ot.partialRight=Yo,Ot.partition=Fo,Ot.pick=Af,Ot.pickBy=zu,Ot.property=Fu,Ot.propertyOf=function(t){return function(n){return null==t?F:Rn(t,n)}},Ot.pull=mo,Ot.pullAll=Pe,Ot.pullAllBy=function(t,n,r){return t&&t.length&&n&&n.length?or(t,n,pe(r,2)):t;
},Ot.pullAllWith=function(t,n,r){return t&&t.length&&n&&n.length?or(t,n,F,r):t},Ot.pullAt=Ao,Ot.range=Zf,Ot.rangeRight=qf,Ot.rearg=Ho,Ot.reject=function(t,n){return(tf(t)?f:kn)(t,iu(pe(n,3)))},Ot.remove=function(t,n){var r=[];if(!t||!t.length)return r;var e=-1,u=[],i=t.length;for(n=pe(n,3);++e<i;){var o=t[e];n(o,e,t)&&(r.push(o),u.push(e))}return fr(t,u),r},Ot.rest=function(t,n){if(typeof t!="function")throw new Qu("Expected a function");return n=n===F?n:wu(n),lr(t,n)},Ot.reverse=Ze,Ot.sampleSize=function(t,n,r){
return n=(r?me(t,n,r):n===F)?1:wu(n),(tf(t)?fn:hr)(t,n)},Ot.set=function(t,n,r){return null==t?t:pr(t,n,r)},Ot.setWith=function(t,n,r,e){return e=typeof e=="function"?e:F,null==t?t:pr(t,n,r,e)},Ot.shuffle=function(t){return(tf(t)?cn:_r)(t)},Ot.slice=function(t,n,r){var e=t?t.length:0;return e?(r&&typeof r!="number"&&me(t,n,r)?(n=0,r=e):(n=null==n?0:wu(n),r=r===F?e:wu(r)),vr(t,n,r)):[]},Ot.sortBy=No,Ot.sortedUniq=function(t){return t&&t.length?br(t):[]},Ot.sortedUniqBy=function(t,n){return t&&t.length?br(t,pe(n,2)):[];
},Ot.split=function(t,n,r){return r&&typeof r!="number"&&me(t,n,r)&&(n=r=F),r=r===F?4294967295:r>>>0,r?(t=Eu(t))&&(typeof n=="string"||null!=n&&!of(n))&&(n=jr(n),!n&&Wt.test(t))?Ir($(t),0,r):t.split(n,r):[]},Ot.spread=function(t,n){if(typeof t!="function")throw new Qu("Expected a function");return n=n===F?0:zi(wu(n),0),lr(function(e){var u=e[n];return e=Ir(e,0,n),u&&s(e,u),r(t,this,e)})},Ot.tail=function(t){var n=t?t.length:0;return n?vr(t,1,n):[]},Ot.take=function(t,n,r){return t&&t.length?(n=r||n===F?1:wu(n),
vr(t,0,0>n?0:n)):[]},Ot.takeRight=function(t,n,r){var e=t?t.length:0;return e?(n=r||n===F?1:wu(n),n=e-n,vr(t,0>n?0:n,e)):[]},Ot.takeRightWhile=function(t,n){return t&&t.length?mr(t,pe(n,3),false,true):[]},Ot.takeWhile=function(t,n){return t&&t.length?mr(t,pe(n,3)):[]},Ot.tap=function(t,n){return n(t),t},Ot.throttle=function(t,n,r){var e=true,u=true;if(typeof t!="function")throw new Qu("Expected a function");return _u(r)&&(e="leading"in r?!!r.leading:e,u="trailing"in r?!!r.trailing:u),eu(t,n,{leading:e,maxWait:n,
trailing:u})},Ot.thru=Ge,Ot.toArray=xu,Ot.toPairs=kf,Ot.toPairsIn=Ef,Ot.toPath=function(t){return tf(t)?l(t,Le):bu(t)?[t]:Cr(vo(t))},Ot.toPlainObject=ku,Ot.transform=function(t,n,r){var e=tf(t)||cf(t);if(n=pe(n,4),null==r)if(e||_u(t)){var i=t.constructor;r=e?tf(t)?new i:[]:su(i)?Xi(_i(t)):{}}else r={};return(e?u:On)(t,function(t,e,u){return n(r,t,e,u)}),r},Ot.unary=function(t){return Xe(t,1)},Ot.union=ko,Ot.unionBy=Eo,Ot.unionWith=Oo,Ot.uniq=function(t){return t&&t.length?wr(t):[]},Ot.uniqBy=function(t,n){
return t&&t.length?wr(t,pe(n,2)):[]},Ot.uniqWith=function(t,n){return t&&t.length?wr(t,F,n):[]},Ot.unset=function(t,n){var r;if(null==t)r=true;else{r=t;var e=n,e=Ae(e,r)?[e]:Sr(e);r=Re(r,e),e=Le(Ne(e)),r=!(null!=r&&ui.call(r,e))||delete r[e]}return r},Ot.unzip=qe,Ot.unzipWith=Ve,Ot.update=function(t,n,r){return null==t?t:pr(t,n,(typeof r=="function"?r:Mu)(Rn(t,n)),void 0)},Ot.updateWith=function(t,n,r,e){return e=typeof e=="function"?e:F,null!=t&&(t=pr(t,n,(typeof r=="function"?r:Mu)(Rn(t,n)),e)),t;
},Ot.values=Wu,Ot.valuesIn=function(t){return null==t?[]:I(t,Ru(t))},Ot.without=So,Ot.words=Uu,Ot.wrap=function(t,n){return n=null==n?Mu:n,Jo(n,t)},Ot.xor=Io,Ot.xorBy=Ro,Ot.xorWith=zo,Ot.zip=Wo,Ot.zipObject=function(t,n){return Er(t||[],n||[],sn)},Ot.zipObjectDeep=function(t,n){return Er(t||[],n||[],pr)},Ot.zipWith=Bo,Ot.entries=kf,Ot.entriesIn=Ef,Ot.extend=hf,Ot.extendWith=pf,Tu(Ot,Ot),Ot.add=Vf,Ot.attempt=Uf,Ot.camelCase=Of,Ot.capitalize=Bu,Ot.ceil=Kf,Ot.clamp=function(t,n,r){return r===F&&(r=n,
n=F),r!==F&&(r=Au(r),r=r===r?r:0),n!==F&&(n=Au(n),n=n===n?n:0),dn(Au(t),n,r)},Ot.clone=function(t){return yn(t,false,true)},Ot.cloneDeep=function(t){return yn(t,true,true)},Ot.cloneDeepWith=function(t,n){return yn(t,true,true,n)},Ot.cloneWith=function(t,n){return yn(t,false,true,n)},Ot.conformsTo=function(t,n){return null==n||xn(t,n,Iu(n))},Ot.deburr=Lu,Ot.defaultTo=function(t,n){return null==t||t!==t?n:t},Ot.divide=Gf,Ot.endsWith=function(t,n,r){t=Eu(t),n=jr(n);var e=t.length,e=r=r===F?e:dn(wu(r),0,e);return r-=n.length,
0<=r&&t.slice(r,e)==n},Ot.eq=ou,Ot.escape=function(t){return(t=Eu(t))&&H.test(t)?t.replace(J,rn):t},Ot.escapeRegExp=function(t){return(t=Eu(t))&&ot.test(t)?t.replace(it,"\\$&"):t},Ot.every=function(t,n,r){var e=tf(t)?o:mn;return r&&me(t,n,r)&&(n=F),e(t,pe(n,3))},Ot.find=Co,Ot.findIndex=De,Ot.findKey=function(t,n){return v(t,pe(n,3),On)},Ot.findLast=Mo,Ot.findLastIndex=Te,Ot.findLastKey=function(t,n){return v(t,pe(n,3),Sn)},Ot.floor=Jf,Ot.forEach=Ye,Ot.forEachRight=He,Ot.forIn=function(t,n){return null==t?t:ro(t,pe(n,3),Ru);
},Ot.forInRight=function(t,n){return null==t?t:eo(t,pe(n,3),Ru)},Ot.forOwn=function(t,n){return t&&On(t,pe(n,3))},Ot.forOwnRight=function(t,n){return t&&Sn(t,pe(n,3))},Ot.get=Ou,Ot.gt=Qo,Ot.gte=Xo,Ot.has=function(t,n){return null!=t&&de(t,n,Bn)},Ot.hasIn=Su,Ot.head=Fe,Ot.identity=Mu,Ot.includes=function(t,n,r,e){return t=cu(t)?t:Wu(t),r=r&&!e?wu(r):0,e=t.length,0>r&&(r=zi(e+r,0)),yu(t)?r<=e&&-1<t.indexOf(n,r):!!e&&-1<d(t,n,r)},Ot.indexOf=function(t,n,r){var e=t?t.length:0;return e?(r=null==r?0:wu(r),
0>r&&(r=zi(e+r,0)),d(t,n,r)):-1},Ot.inRange=function(t,n,r){return n=ju(n),r===F?(r=n,n=0):r=ju(r),t=Au(t),t>=Wi(n,r)&&t<zi(n,r)},Ot.invoke=xf,Ot.isArguments=fu,Ot.isArray=tf,Ot.isArrayBuffer=nf,Ot.isArrayLike=cu,Ot.isArrayLikeObject=au,Ot.isBoolean=function(t){return true===t||false===t||vu(t)&&"[object Boolean]"==fi.call(t)},Ot.isBuffer=rf,Ot.isDate=ef,Ot.isElement=function(t){return null!=t&&1===t.nodeType&&vu(t)&&!du(t)},Ot.isEmpty=function(t){if(cu(t)&&(tf(t)||typeof t=="string"||typeof t.splice=="function"||rf(t)||fu(t)))return!t.length;
var n=Et(t);if("[object Map]"==n||"[object Set]"==n)return!t.size;if(Ee(t))return!Ri(t).length;for(var r in t)if(ui.call(t,r))return false;return true},Ot.isEqual=function(t,n){return $n(t,n)},Ot.isEqualWith=function(t,n,r){var e=(r=typeof r=="function"?r:F)?r(t,n):F;return e===F?$n(t,n,r):!!e},Ot.isError=lu,Ot.isFinite=function(t){return typeof t=="number"&&Si(t)},Ot.isFunction=su,Ot.isInteger=hu,Ot.isLength=pu,Ot.isMap=uf,Ot.isMatch=function(t,n){return t===n||Nn(t,n,ve(n))},Ot.isMatchWith=function(t,n,r){
return r=typeof r=="function"?r:F,Nn(t,n,ve(n),r)},Ot.isNaN=function(t){return gu(t)&&t!=+t},Ot.isNative=function(t){if(so(t))throw new Vu("Unsupported core-js use. Try https://github.com/es-shims.");return Pn(t)},Ot.isNil=function(t){return null==t},Ot.isNull=function(t){return null===t},Ot.isNumber=gu,Ot.isObject=_u,Ot.isObjectLike=vu,Ot.isPlainObject=du,Ot.isRegExp=of,Ot.isSafeInteger=function(t){return hu(t)&&-9007199254740991<=t&&9007199254740991>=t},Ot.isSet=ff,Ot.isString=yu,Ot.isSymbol=bu,
Ot.isTypedArray=cf,Ot.isUndefined=function(t){return t===F},Ot.isWeakMap=function(t){return vu(t)&&"[object WeakMap]"==Et(t)},Ot.isWeakSet=function(t){return vu(t)&&"[object WeakSet]"==fi.call(t)},Ot.join=function(t,n){return t?Ii.call(t,n):""},Ot.kebabCase=Sf,Ot.last=Ne,Ot.lastIndexOf=function(t,n,r){var e=t?t.length:0;if(!e)return-1;var u=e;if(r!==F&&(u=wu(r),u=0>u?zi(e+u,0):Wi(u,e-1)),n===n){for(r=u+1;r--&&t[r]!==n;);t=r}else t=g(t,b,u,true);return t},Ot.lowerCase=If,Ot.lowerFirst=Rf,Ot.lt=af,Ot.lte=lf,
Ot.max=function(t){return t&&t.length?An(t,Mu,Wn):F},Ot.maxBy=function(t,n){return t&&t.length?An(t,pe(n,2),Wn):F},Ot.mean=function(t){return x(t,Mu)},Ot.meanBy=function(t,n){return x(t,pe(n,2))},Ot.min=function(t){return t&&t.length?An(t,Mu,Yn):F},Ot.minBy=function(t,n){return t&&t.length?An(t,pe(n,2),Yn):F},Ot.stubArray=Nu,Ot.stubFalse=Pu,Ot.stubObject=function(){return{}},Ot.stubString=function(){return""},Ot.stubTrue=function(){return true},Ot.multiply=Yf,Ot.nth=function(t,n){return t&&t.length?nr(t,wu(n)):F;
},Ot.noConflict=function(){return Pt._===this&&(Pt._=ci),this},Ot.noop=$u,Ot.now=Po,Ot.pad=function(t,n,r){t=Eu(t);var e=(n=wu(n))?T(t):0;return!n||e>=n?t:(n=(n-e)/2,te(ki(n),r)+t+te(Ai(n),r))},Ot.padEnd=function(t,n,r){t=Eu(t);var e=(n=wu(n))?T(t):0;return n&&e<n?t+te(n-e,r):t},Ot.padStart=function(t,n,r){t=Eu(t);var e=(n=wu(n))?T(t):0;return n&&e<n?te(n-e,r)+t:t},Ot.parseInt=function(t,n,r){return r||null==n?n=0:n&&(n=+n),Li(Eu(t).replace(ct,""),n||0)},Ot.random=function(t,n,r){if(r&&typeof r!="boolean"&&me(t,n,r)&&(n=r=F),
r===F&&(typeof n=="boolean"?(r=n,n=F):typeof t=="boolean"&&(r=t,t=F)),t===F&&n===F?(t=0,n=1):(t=ju(t),n===F?(n=t,t=0):n=ju(n)),t>n){var e=t;t=n,n=e}return r||t%1||n%1?(r=Ui(),Wi(t+r*(n-t+Tt("1e-"+((r+"").length-1))),n)):cr(t,n)},Ot.reduce=function(t,n,r){var e=tf(t)?h:m,u=3>arguments.length;return e(t,pe(n,4),r,u,to)},Ot.reduceRight=function(t,n,r){var e=tf(t)?p:m,u=3>arguments.length;return e(t,pe(n,4),r,u,no)},Ot.repeat=function(t,n,r){return n=(r?me(t,n,r):n===F)?1:wu(n),ar(Eu(t),n)},Ot.replace=function(){
var t=arguments,n=Eu(t[0]);return 3>t.length?n:n.replace(t[1],t[2])},Ot.result=function(t,n,r){n=Ae(n,t)?[n]:Sr(n);var e=-1,u=n.length;for(u||(t=F,u=1);++e<u;){var i=null==t?F:t[Le(n[e])];i===F&&(e=u,i=r),t=su(i)?i.call(t):i}return t},Ot.round=Hf,Ot.runInContext=w,Ot.sample=function(t){return(tf(t)?on:sr)(t)},Ot.size=function(t){if(null==t)return 0;if(cu(t))return yu(t)?T(t):t.length;var n=Et(t);return"[object Map]"==n||"[object Set]"==n?t.size:Gn(t).length},Ot.snakeCase=zf,Ot.some=function(t,n,r){
var e=tf(t)?_:gr;return r&&me(t,n,r)&&(n=F),e(t,pe(n,3))},Ot.sortedIndex=function(t,n){return dr(t,n)},Ot.sortedIndexBy=function(t,n,r){return yr(t,n,pe(r,2))},Ot.sortedIndexOf=function(t,n){var r=t?t.length:0;if(r){var e=dr(t,n);if(e<r&&ou(t[e],n))return e}return-1},Ot.sortedLastIndex=function(t,n){return dr(t,n,true)},Ot.sortedLastIndexBy=function(t,n,r){return yr(t,n,pe(r,2),true)},Ot.sortedLastIndexOf=function(t,n){if(t&&t.length){var r=dr(t,n,true)-1;if(ou(t[r],n))return r}return-1},Ot.startCase=Wf,
Ot.startsWith=function(t,n,r){return t=Eu(t),r=dn(wu(r),0,t.length),n=jr(n),t.slice(r,r+n.length)==n},Ot.subtract=Qf,Ot.sum=function(t){return t&&t.length?k(t,Mu):0},Ot.sumBy=function(t,n){return t&&t.length?k(t,pe(n,2)):0},Ot.template=function(t,n,r){var e=Ot.templateSettings;r&&me(t,n,r)&&(n=F),t=Eu(t),n=pf({},n,e,an),r=pf({},n.imports,e.imports,an);var u,i,o=Iu(r),f=I(r,o),c=0;r=n.interpolate||mt;var a="__p+='";r=Yu((n.escape||mt).source+"|"+r.source+"|"+(r===tt?vt:mt).source+"|"+(n.evaluate||mt).source+"|$","g");
var l="sourceURL"in n?"//# sourceURL="+n.sourceURL+"\n":"";if(t.replace(r,function(n,r,e,o,f,l){return e||(e=o),a+=t.slice(c,l).replace(At,B),r&&(u=true,a+="'+__e("+r+")+'"),f&&(i=true,a+="';"+f+";\n__p+='"),e&&(a+="'+((__t=("+e+"))==null?'':__t)+'"),c=l+n.length,n}),a+="';",(n=n.variable)||(a="with(obj){"+a+"}"),a=(i?a.replace(q,""):a).replace(V,"$1").replace(K,"$1;"),a="function("+(n||"obj")+"){"+(n?"":"obj||(obj={});")+"var __t,__p=''"+(u?",__e=_.escape":"")+(i?",__j=Array.prototype.join;function print(){__p+=__j.call(arguments,'')}":";")+a+"return __p}",
n=Uf(function(){return Ku(o,l+"return "+a).apply(F,f)}),n.source=a,lu(n))throw n;return n},Ot.times=function(t,n){if(t=wu(t),1>t||9007199254740991<t)return[];var r=4294967295,e=Wi(t,4294967295);for(n=pe(n),t-=4294967295,e=E(e,n);++r<t;)n(r);return e},Ot.toFinite=ju,Ot.toInteger=wu,Ot.toLength=mu,Ot.toLower=function(t){return Eu(t).toLowerCase()},Ot.toNumber=Au,Ot.toSafeInteger=function(t){return dn(wu(t),-9007199254740991,9007199254740991)},Ot.toString=Eu,Ot.toUpper=function(t){return Eu(t).toUpperCase();
},Ot.trim=function(t,n,r){return(t=Eu(t))&&(r||n===F)?t.replace(ft,""):t&&(n=jr(n))?(t=$(t),r=$(n),n=z(t,r),r=W(t,r)+1,Ir(t,n,r).join("")):t},Ot.trimEnd=function(t,n,r){return(t=Eu(t))&&(r||n===F)?t.replace(at,""):t&&(n=jr(n))?(t=$(t),n=W(t,$(n))+1,Ir(t,0,n).join("")):t},Ot.trimStart=function(t,n,r){return(t=Eu(t))&&(r||n===F)?t.replace(ct,""):t&&(n=jr(n))?(t=$(t),n=z(t,$(n)),Ir(t,n).join("")):t},Ot.truncate=function(t,n){var r=30,e="...";if(_u(n))var u="separator"in n?n.separator:u,r="length"in n?wu(n.length):r,e="omission"in n?jr(n.omission):e;
t=Eu(t);var i=t.length;if(Wt.test(t))var o=$(t),i=o.length;if(r>=i)return t;if(i=r-T(e),1>i)return e;if(r=o?Ir(o,0,i).join(""):t.slice(0,i),u===F)return r+e;if(o&&(i+=r.length-i),of(u)){if(t.slice(i).search(u)){var f=r;for(u.global||(u=Yu(u.source,Eu(gt.exec(u))+"g")),u.lastIndex=0;o=u.exec(f);)var c=o.index;r=r.slice(0,c===F?i:c)}}else t.indexOf(jr(u),i)!=i&&(u=r.lastIndexOf(u),-1<u&&(r=r.slice(0,u)));return r+e},Ot.unescape=function(t){return(t=Eu(t))&&Y.test(t)?t.replace(G,en):t},Ot.uniqueId=function(t){
var n=++ii;return Eu(t)+n},Ot.upperCase=Bf,Ot.upperFirst=Lf,Ot.each=Ye,Ot.eachRight=He,Ot.first=Fe,Tu(Ot,function(){var t={};return On(Ot,function(n,r){ui.call(Ot.prototype,r)||(t[r]=n)}),t}(),{chain:false}),Ot.VERSION="4.16.3",u("bind bindKey curry curryRight partial partialRight".split(" "),function(t){Ot[t].placeholder=Ot}),u(["drop","take"],function(t,n){Dt.prototype[t]=function(r){var e=this.__filtered__;if(e&&!n)return new Dt(this);r=r===F?1:zi(wu(r),0);var u=this.clone();return e?u.__takeCount__=Wi(r,u.__takeCount__):u.__views__.push({
size:Wi(r,4294967295),type:t+(0>u.__dir__?"Right":"")}),u},Dt.prototype[t+"Right"]=function(n){return this.reverse()[t](n).reverse()}}),u(["filter","map","takeWhile"],function(t,n){var r=n+1,e=1==r||3==r;Dt.prototype[t]=function(t){var n=this.clone();return n.__iteratees__.push({iteratee:pe(t,3),type:r}),n.__filtered__=n.__filtered__||e,n}}),u(["head","last"],function(t,n){var r="take"+(n?"Right":"");Dt.prototype[t]=function(){return this[r](1).value()[0]}}),u(["initial","tail"],function(t,n){var r="drop"+(n?"":"Right");
Dt.prototype[t]=function(){return this.__filtered__?new Dt(this):this[r](1)}}),Dt.prototype.compact=function(){return this.filter(Mu)},Dt.prototype.find=function(t){return this.filter(t).head()},Dt.prototype.findLast=function(t){return this.reverse().find(t)},Dt.prototype.invokeMap=lr(function(t,n){return typeof t=="function"?new Dt(this):this.map(function(r){return Mn(r,t,n)})}),Dt.prototype.reject=function(t){return this.filter(iu(pe(t)))},Dt.prototype.slice=function(t,n){t=wu(t);var r=this;return r.__filtered__&&(0<t||0>n)?new Dt(r):(0>t?r=r.takeRight(-t):t&&(r=r.drop(t)),
n!==F&&(n=wu(n),r=0>n?r.dropRight(-n):r.take(n-t)),r)},Dt.prototype.takeRightWhile=function(t){return this.reverse().takeWhile(t).reverse()},Dt.prototype.toArray=function(){return this.take(4294967295)},On(Dt.prototype,function(t,n){var r=/^(?:filter|find|map|reject)|While$/.test(n),e=/^(?:head|last)$/.test(n),u=Ot[e?"take"+("last"==n?"Right":""):n],i=e||/^find/.test(n);u&&(Ot.prototype[n]=function(){function n(t){return t=u.apply(Ot,s([t],f)),e&&h?t[0]:t}var o=this.__wrapped__,f=e?[1]:arguments,c=o instanceof Dt,a=f[0],l=c||tf(o);
l&&r&&typeof a=="function"&&1!=a.length&&(c=l=false);var h=this.__chain__,p=!!this.__actions__.length,a=i&&!h,c=c&&!p;return!i&&l?(o=c?o:new Dt(this),o=t.apply(o,f),o.__actions__.push({func:Ge,args:[n],thisArg:F}),new Mt(o,h)):a&&c?t.apply(this,f):(o=this.thru(n),a?e?o.value()[0]:o.value():o)})}),u("pop push shift sort splice unshift".split(" "),function(t){var n=Xu[t],r=/^(?:push|sort|unshift)$/.test(t)?"tap":"thru",e=/^(?:pop|shift)$/.test(t);Ot.prototype[t]=function(){var t=arguments;if(e&&!this.__chain__){
var u=this.value();return n.apply(tf(u)?u:[],t)}return this[r](function(r){return n.apply(tf(r)?r:[],t)})}}),On(Dt.prototype,function(t,n){var r=Ot[n];if(r){var e=r.name+"";(Zi[e]||(Zi[e]=[])).push({name:n,func:r})}}),Zi[Yr(F,2).name]=[{name:"wrapper",func:F}],Dt.prototype.clone=function(){var t=new Dt(this.__wrapped__);return t.__actions__=Cr(this.__actions__),t.__dir__=this.__dir__,t.__filtered__=this.__filtered__,t.__iteratees__=Cr(this.__iteratees__),t.__takeCount__=this.__takeCount__,t.__views__=Cr(this.__views__),
t},Dt.prototype.reverse=function(){if(this.__filtered__){var t=new Dt(this);t.__dir__=-1,t.__filtered__=true}else t=this.clone(),t.__dir__*=-1;return t},Dt.prototype.value=function(){var t,n=this.__wrapped__.value(),r=this.__dir__,e=tf(n),u=0>r,i=e?n.length:0;t=i;for(var o=this.__views__,f=0,c=-1,a=o.length;++c<a;){var l=o[c],s=l.size;switch(l.type){case"drop":f+=s;break;case"dropRight":t-=s;break;case"take":t=Wi(t,f+s);break;case"takeRight":f=zi(f,t-s)}}if(t={start:f,end:t},o=t.start,f=t.end,t=f-o,
u=u?f:o-1,o=this.__iteratees__,f=o.length,c=0,a=Wi(t,this.__takeCount__),!e||200>i||i==t&&a==t)return Ar(n,this.__actions__);e=[];t:for(;t--&&c<a;){for(u+=r,i=-1,l=n[u];++i<f;){var h=o[i],s=h.type,h=(0,h.iteratee)(l);if(2==s)l=h;else if(!h){if(1==s)continue t;break t}}e[c++]=l}return e},Ot.prototype.at=Lo,Ot.prototype.chain=function(){return Ke(this)},Ot.prototype.commit=function(){return new Mt(this.value(),this.__chain__)},Ot.prototype.next=function(){this.__values__===F&&(this.__values__=xu(this.value()));
var t=this.__index__>=this.__values__.length;return{done:t,value:t?F:this.__values__[this.__index__++]}},Ot.prototype.plant=function(t){for(var n,r=this;r instanceof Rt;){var e=Me(r);e.__index__=0,e.__values__=F,n?u.__wrapped__=e:n=e;var u=e,r=r.__wrapped__}return u.__wrapped__=t,n},Ot.prototype.reverse=function(){var t=this.__wrapped__;return t instanceof Dt?(this.__actions__.length&&(t=new Dt(this)),t=t.reverse(),t.__actions__.push({func:Ge,args:[Ze],thisArg:F}),new Mt(t,this.__chain__)):this.thru(Ze);
},Ot.prototype.toJSON=Ot.prototype.valueOf=Ot.prototype.value=function(){return Ar(this.__wrapped__,this.__actions__)},Ot.prototype.first=Ot.prototype.head,vi&&(Ot.prototype[vi]=Je),Ot}();typeof define=="function"&&typeof define.amd=="object"&&define.amd?(Pt._=un, define(function(){return un})):qt?((qt.exports=un)._=un,Zt._=un):Pt._=un}).call(this);
<!DOCTYPE html>
<meta charset='utf-8'>
<link rel="stylesheet" type="text/css" href="style.css">
<body></body>
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/r83/three.min.js"></script>
<script src='d3v4.js'></script>
<script src='_script.js'></script>
{
"name": "2140",
"version": "1.0.0",
"description": "Early/simplifed version of [You Draw It: What Got Better or Worse During Obama’s Presidency](http://www.nytimes.com/interactive/2017/01/15/us/politics/you-draw-obama-legacy.html)",
"main": "_script.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"cheerio": "^0.22.0",
"d3": "^4.7.3",
"glob": "^7.1.1",
"underscore": "^1.8.3"
}
}
var glob = require('glob')
var d3 = require('d3')
var fs = require('fs')
var cheerio = require('cheerio')
var _ = require('underscore')
var chapters = []
glob.sync('OEBPS/*.xhtml').forEach(function(path){
if (!path.includes('chapter')) return
var chapter = +path.slice(13, 16)
// if (chapter > 50) return
var $ = cheerio.load(fs.readFileSync(path, 'utf-8'))
var text = $('p').map(function(){
return $(this).text()
}).get().join('.')
var name = $('.chapter-number').text().slice(3, 100)
// console.log(path, chapter, name)
chapters.push({chapter, name, text, length: text.length})
})
// var names = _.uniq(chapters.map(d => d.name))
// console.log(names)
var names = [
'Mutt',
'Jeff',
'Gen',
'Franklin',
'Vlade',
'Citizen',
'Amelia',
'Charlotte',
'Stefan',
'Roberto'
]
chapters.forEach(chapter => {
if (chapter.name.includes(' cit')) chapter.name = 'Citizen'
chapter.links = []
names.forEach(name => {
var re = new RegExp(`[^.]*${name}[^.]*`, 'g')
while ((match = re.exec(chapter.text)) != null) {
chapter.links.push({index: match.index, str: match[0].trim(), name})
}
})
delete chapter.text
})
fs.writeFileSync('chapters.json', JSON.stringify(chapters))
svg{
overflow: visible;
}
body{
font-family: monospace;
margin: 0px;
overflow: hidden;
}
.axis{
opacity: .5;
}
.line{
fill: none;
stroke: black;
stroke-width: 3;
}
.area{
fill: #eee;
}
.your-line{
stroke: #f0f;
stroke-width: 3;
stroke-dasharray: 5 5;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment