|
@@ -2,45 +2,69 @@
|
|
<svg
|
|
<svg
|
|
width="100%" height="100%"
|
|
width="100%" height="100%"
|
|
ref="svg" :id="id"
|
|
ref="svg" :id="id"
|
|
|
|
+ :view-box.camel="viewBox"
|
|
>
|
|
>
|
|
- <template v-if="ready">
|
|
|
|
- <path
|
|
|
|
- v-for="(item, index) in links"
|
|
|
|
- :key="`link_${index}`"
|
|
|
|
- :d="lineGenerator([item.source, item.target])"
|
|
|
|
- :class="['link', item.linkType || item.target.data.linkType]"
|
|
|
|
|
|
+ <g :id="id + '-map'" v-if="ready">
|
|
|
|
+ <g>
|
|
|
|
+ <path
|
|
|
|
+ v-for="(item, index) in links"
|
|
|
|
+ :key="`link_${index}`"
|
|
|
|
+ :d="lineGenerator([item.source, item.target])"
|
|
|
|
+ :class="['link', item.linkType || item.target.data.linkType]"
|
|
|
|
+ />
|
|
|
|
+ </g>
|
|
|
|
+
|
|
|
|
+ <circle
|
|
|
|
+ v-for="(node, index) in nodes"
|
|
|
|
+ :key="'circle' + index"
|
|
|
|
+ :cx="node.x"
|
|
|
|
+ :cy="node.y"
|
|
|
|
+ class="svg-dot-btn"
|
|
|
|
+ :class="['svg-dot-btn-' + node.data.variant, { 'origin': node.data.isOrigin }]"
|
|
|
|
+ tabindex="0"
|
|
|
|
+ @click="onNodeClick(node.data)"
|
|
|
|
+ @keyup.enter="onNodeClick(node.data)"
|
|
|
|
+ @mouseenter="activeNode = node"
|
|
/>
|
|
/>
|
|
|
|
|
|
- <template v-for="(node, index) in nodes">
|
|
|
|
- <circle
|
|
|
|
- :key="'circle' + index"
|
|
|
|
- :cx="node.x"
|
|
|
|
- :cy="node.y"
|
|
|
|
- :class="node.data.class + (node.data.first ? ' first' : '')"
|
|
|
|
- :r="node.children ? 7 : 5"
|
|
|
|
- tabindex="0"
|
|
|
|
- @click="onNodeClick(node.data)"
|
|
|
|
- @keyup.enter="onNodeClick(node.data)"
|
|
|
|
|
|
+ <g>
|
|
|
|
+ <!-- rect for forcing full render -->
|
|
|
|
+ <rect
|
|
|
|
+ x="-50%" y="-50%" fill="transparent"
|
|
|
|
+ width="100%" height="100%" style="pointer-events: none;"
|
|
|
|
+ />
|
|
|
|
+ <foreignObject
|
|
|
|
+ v-if="activeNode"
|
|
|
|
+ :x="activeNode.x"
|
|
|
|
+ :y="activeNode.y"
|
|
>
|
|
>
|
|
- <title>{{ node.data.title }}</title>
|
|
|
|
- </circle>
|
|
|
|
-
|
|
|
|
- <text
|
|
|
|
- v-if="showId"
|
|
|
|
- :key="'text' + index" :x="node.x + 7" :y="node.y-7"
|
|
|
|
- >{{ node.data.id }}</text>
|
|
|
|
- </template>
|
|
|
|
- </template>
|
|
|
|
|
|
+ <dot-button
|
|
|
|
+ :variant="activeNode.data.variant"
|
|
|
|
+ class="active"
|
|
|
|
+ @click="onNodeClick(activeNode.data)"
|
|
|
|
+ @mouseleave="activeNode = null"
|
|
|
|
+ >
|
|
|
|
+ <template v-if="activeNode.data.variant !== 'depart'">
|
|
|
|
+ {{ activeNode.data.families[0].name }}<br>
|
|
|
|
+ </template>
|
|
|
|
+ {{ toCommaList(activeNode.data.authors) }},<br>
|
|
|
|
+ <span v-if="activeNode.data.field_titre_regular" v-html="activeNode.data.field_titre_regular + ','" />
|
|
|
|
+ <span v-html="activeNode.data.field_titre_italique || '<em>pas de titre ital</em>'" />
|
|
|
|
+ </dot-button>
|
|
|
|
+ </foreignObject>
|
|
|
|
+ </g>
|
|
|
|
+ </g>
|
|
</svg>
|
|
</svg>
|
|
</template>
|
|
</template>
|
|
|
|
|
|
<script>
|
|
<script>
|
|
import { line } from 'd3-shape'
|
|
import { line } from 'd3-shape'
|
|
import { selectAll } from 'd3-selection'
|
|
import { selectAll } from 'd3-selection'
|
|
|
|
+import { select, zoom } from 'd3'
|
|
import { drag } from 'd3-drag'
|
|
import { drag } from 'd3-drag'
|
|
import {
|
|
import {
|
|
forceSimulation, forceLink, forceManyBody,
|
|
forceSimulation, forceLink, forceManyBody,
|
|
- forceX, forceY, forceCenter
|
|
|
|
|
|
+ forceX, forceY
|
|
} from 'd3-force'
|
|
} from 'd3-force'
|
|
|
|
|
|
|
|
|
|
@@ -54,8 +78,7 @@ export default {
|
|
props: {
|
|
props: {
|
|
nodes: { type: Array, required: true },
|
|
nodes: { type: Array, required: true },
|
|
links: { type: Array, required: true },
|
|
links: { type: Array, required: true },
|
|
- id: { type: String, default: 'node-map' },
|
|
|
|
- showId: { type: Boolean, default: true }
|
|
|
|
|
|
+ id: { type: String, default: 'node-map' }
|
|
},
|
|
},
|
|
|
|
|
|
data () {
|
|
data () {
|
|
@@ -64,11 +87,28 @@ export default {
|
|
width: 100,
|
|
width: 100,
|
|
height: 100,
|
|
height: 100,
|
|
simulation: forceSimulation(),
|
|
simulation: forceSimulation(),
|
|
- lineGenerator: line().x(node => node.x).y(node => node.y)
|
|
|
|
|
|
+ lineGenerator: line().x(node => node.x).y(node => node.y),
|
|
|
|
+ activeNode: null
|
|
|
|
+ }
|
|
|
|
+ },
|
|
|
|
+
|
|
|
|
+ computed: {
|
|
|
|
+ viewBox () {
|
|
|
|
+ const { width, height } = this
|
|
|
|
+ return `-${width / 2} -${height / 2} ${width} ${height}`
|
|
}
|
|
}
|
|
},
|
|
},
|
|
|
|
|
|
methods: {
|
|
methods: {
|
|
|
|
+ toCommaList (arr) {
|
|
|
|
+ // FIXME TEMP some texts doesn't have authors
|
|
|
|
+ try {
|
|
|
|
+ return arr.map(({ name }) => name).join(', ')
|
|
|
|
+ } catch {
|
|
|
|
+ return 'pas d\'auteur⋅rices'
|
|
|
|
+ }
|
|
|
|
+ },
|
|
|
|
+
|
|
updateSize () {
|
|
updateSize () {
|
|
const { width, height } = this.$el.getBoundingClientRect()
|
|
const { width, height } = this.$el.getBoundingClientRect()
|
|
Object.assign(this.$data, { width, height })
|
|
Object.assign(this.$data, { width, height })
|
|
@@ -90,6 +130,14 @@ export default {
|
|
|
|
|
|
initListeners () {
|
|
initListeners () {
|
|
// TODO replace with vue events ?
|
|
// TODO replace with vue events ?
|
|
|
|
+ const svg = select('#' + this.id)
|
|
|
|
+ const map = select('#' + this.id + '-map')
|
|
|
|
+ svg.call(zoom()
|
|
|
|
+ .extent([[0, 0], [this.width, this.height]])
|
|
|
|
+ .scaleExtent([0.3, 1])
|
|
|
|
+ .on('zoom', ({ transform }) => {
|
|
|
|
+ map.attr('transform', transform)
|
|
|
|
+ }))
|
|
|
|
|
|
selectAll(`#${this.id} circle`)
|
|
selectAll(`#${this.id} circle`)
|
|
.data(this.nodes)
|
|
.data(this.nodes)
|
|
@@ -103,11 +151,18 @@ export default {
|
|
setupForces () {
|
|
setupForces () {
|
|
this.simulation
|
|
this.simulation
|
|
.nodes(this.nodes)
|
|
.nodes(this.nodes)
|
|
- .force('link', forceLink(this.links).id(d => d.id).distance(50))
|
|
|
|
- .force('charge', forceManyBody().strength(-150))
|
|
|
|
|
|
+ .force('charge', forceManyBody().strength((d) => {
|
|
|
|
+ if (d.data.linkType === 'parents') return -2000
|
|
|
|
+ if (d.data.linkType === 'siblings') return -500
|
|
|
|
+ return -1000
|
|
|
|
+ }))
|
|
|
|
+ .force('link', forceLink(this.links).id(d => d.id).distance((d) => {
|
|
|
|
+ if (d.linkType === 'parents') return 200
|
|
|
|
+ if (d.linkType === 'siblings') return 100
|
|
|
|
+ return 0
|
|
|
|
+ }).strength(0.5))
|
|
.force('x', forceX())
|
|
.force('x', forceX())
|
|
.force('y', forceY())
|
|
.force('y', forceY())
|
|
- .force('center', forceCenter(this.width * 0.5, this.height * 0.5))
|
|
|
|
},
|
|
},
|
|
|
|
|
|
// D3 DRAG EVENT HANDLERS
|
|
// D3 DRAG EVENT HANDLERS
|
|
@@ -145,7 +200,7 @@ export default {
|
|
}
|
|
}
|
|
},
|
|
},
|
|
|
|
|
|
- async mounted () {
|
|
|
|
|
|
+ mounted () {
|
|
this.init()
|
|
this.init()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
@@ -154,6 +209,7 @@ export default {
|
|
<style lang="scss" scoped>
|
|
<style lang="scss" scoped>
|
|
.link {
|
|
.link {
|
|
stroke: grey;
|
|
stroke: grey;
|
|
|
|
+ vector-effect: non-scaling-stroke;
|
|
|
|
|
|
&.siblings {
|
|
&.siblings {
|
|
stroke-dasharray: 4;
|
|
stroke-dasharray: 4;
|
|
@@ -165,21 +221,34 @@ export default {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-text {
|
|
|
|
- font-size: 0.7rem;
|
|
|
|
- user-select: none;
|
|
|
|
-}
|
|
|
|
|
|
+.svg-dot-btn {
|
|
|
|
+ r: 9.5px;
|
|
|
|
|
|
-@each $id in map-keys($families){
|
|
|
|
- .family-#{$id} {
|
|
|
|
- fill: setColorFromId($id);
|
|
|
|
- @if $id == 9 {
|
|
|
|
- stroke: black;
|
|
|
|
|
|
+ @each $color, $value in $theme-colors {
|
|
|
|
+ &-#{$color} {
|
|
|
|
+ fill: $value;
|
|
|
|
+
|
|
|
|
+ @if $color == 'depart' {
|
|
|
|
+ stroke: $black;
|
|
|
|
+ stroke-width: 3px;
|
|
|
|
+ vector-effect: non-scaling-stroke;
|
|
|
|
+ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ &.origin {
|
|
|
|
+ r: 15px;
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+foreignObject {
|
|
|
|
+ height: 1px;
|
|
|
|
+ width: 1px;
|
|
|
|
+ overflow: visible;
|
|
}
|
|
}
|
|
|
|
|
|
-.first {
|
|
|
|
- stroke-width: 2px;
|
|
|
|
|
|
+.dot-btn {
|
|
|
|
+ vertical-align: top;
|
|
|
|
+ transform: translate(-50%, -50%);
|
|
}
|
|
}
|
|
</style>
|
|
</style>
|