File size: 8,333 Bytes
63858e7
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
fe4a287
63858e7
fe4a287
63858e7
fe4a287
63858e7
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
fe4a287
63858e7
 
 
 
 
 
 
 
 
 
fe4a287
63858e7
 
 
 
 
 
 
 
 
 
fe4a287
63858e7
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
import * as d3 from "d3";
import 'd3-selection-multi'
import { D3Sel } from "../etc/Util";
import { Edge, EdgeData } from "./EdgeConnector"
import { VComponent } from "./VisComponent";
import { SimpleEventHandler } from "../etc/SimpleEventHandler";
import * as tp from "../etc/types"

export type AttentionData = number[][]

export const scaleLinearWidth = opacity => 5 * opacity^0.33;

export class AttentionGraph extends VComponent<AttentionData>{
    css_name = '';
    _current: {};

    _data: AttentionData; // The passed data
    edgeData: EdgeData; // A wrapper around _data. User should not mind
    plotData: Edge[]; // Needed for plotting

    /** COMPONENTS
     * Expose the components belonging to the class as properties of the class. 
     * This is useful to create methods that specifically modify a single part or component without having to reselect it. 
     * Makes for more responsive applications
     * */
    svg: D3Sel;
    graph: D3Sel;

    // The below components require data
    paths: D3Sel;
    opacityScales: d3.ScaleLinear<any, any>[];
    linkGen: d3.Link<any, any, any>

    // OPTIONS WITH DEFAULTS
    _threshold = 0.7; // Accumulation threshold. Between 0-1
    normBy: tp.NormBy

    static events = {} // No events needed for this one

    options = {
        boxheight: 26, // The height of the div boxes around the SVG element
        height: 500,
        width: 200,
        offset: 0, // Should I offset the left side by 1 or not?
    }

    constructor(d3Parent: D3Sel, eventHandler?: SimpleEventHandler, options: {} = {}) {
        super(d3Parent, eventHandler)
        this.superInitSVG(options)
        this._init()
    }

    _init() {
        this.svg = this.parent;
        this.graph = this.svg.selectAll(`.atn-curve`);
        this.linkGen = d3.linkHorizontal()
            .x(d => d[0])
            .y(d => d[1]);
    }

    // Define whether to use the 'j' or 'i' attribute to calculate opacities
    private scaleIdx(): "i" | "j" {
        switch (this.normBy) {
            case tp.NormBy.COL:
                return 'j'
            case tp.NormBy.ROW:
                return 'i'
            case tp.NormBy.ALL:
                return 'i'

        }

    }

    /**
     * Create connections between locations of the SVG using D3's linkGen
     */
    private createConnections() {
        const self = this;
        const op = this.options;
        if (this.paths) {
            this.paths.attrs({
                'd': (d, i) => {
                    const data: { source: [number, number], target: [number, number] } =
                    {
                        source: [0, op.boxheight * (d.i + 0.5 + op.offset)],
                        target: [op.width, op.boxheight * (d.j + 0.5)] // + 2 allows small offset
                    };
                    return this.linkGen(data);
                },
                'class': 'atn-curve'
            })
                .attr("src-idx", (d, i) => d.i)
                .attr("target-idx", (d, i) => d.j);
        }
    }

    /**
     * Change the height of the SVG
     */
    private updateHeight() {
        const op = this.options;
        if (this.svg != null) {
            this.svg.attr("height", this.options.height + (op.offset * this.options.boxheight))
        }
        return this;
    }

    /**
     * Change the width of the SVG
     */
    private updateWidth() {
        if (this.svg != null) {
            this.svg.attr("width", this.options.width)
        }
        return this;
    }

    /**
     * Change the Opacity of the lines according to the value of the data
     */
    private updateOpacity() {
        const self = this;
        if (this.paths != null) {
            // paths.transition().duration(500).attr('opacity', (d) => {
            this.paths.attr('opacity', (d) => {
                const val = this.opacityScales[d[self.scaleIdx()]](d.v);
                return val;
            })
            this.paths.attr('stroke-width', (d) => {
                const val = this.opacityScales[d[self.scaleIdx()]](d.v);
                return scaleLinearWidth(val) //5 * val^0.33;
            })
        }
        return this;
    }

    /**
     * Rerender the graph in the event that the data changes
     */
    private updateData() {
        if (this.graph != null) {
            d3.selectAll(".atn-curve").remove();

            const data = this.plotData

            this.paths = this.graph
                .data(data)
                .join('path');

            this.createConnections();
            this.updateOpacity();

            return this;
        }
    }

    /**
     * Scale the opacity according to the values of the data, from 0 to max of contained data
     * Normalize by each source target, or across the whole
     */
    private createScales = () => {
        this.opacityScales = [];
        let arr = []

        // Group normalization
        switch (this.normBy){
            case tp.NormBy.ROW:
                arr = this.edgeData.extent(1);
                this.opacityScales = [];
                arr.forEach((v, i) => {
                    (this.opacityScales as d3.ScaleLinear<any, any>[]).push(
                        d3.scaleLinear()
                            .domain([0, v[1]])
                            .range([0, 0.9])
                    )
                })
                break;
            case tp.NormBy.COL:
                arr = this.edgeData.extent(0);
                this.opacityScales = [];
                arr.forEach((v, i) => {
                    (this.opacityScales as d3.ScaleLinear<any, any>[]).push(
                        d3.scaleLinear()
                            .domain([0, v[1]])
                            .range([0, 0.9])
                    )
                })
                break;
            case tp.NormBy.ALL:
                const maxIn = d3.max(this.plotData.map((d) => d.v))
                for (let i = 0; i < this._data.length; i++) {
                    this.opacityScales.push(d3.scaleLinear()
                        .domain([0, maxIn])
                        .range([0, 1]));
                }
                break;
            default:
                console.log("Nor norming specified");
                break;
        }
    }

    /**
     * Access / modify the data in a D3 style way. If modified, the component will update just the part that is needed to be updated
     */
    data(): AttentionData
    data(value: AttentionData): this
    data(value?) {
        if (value == null) {
            return this._data;
        }

        this._data = value;
        this.edgeData = new EdgeData(value);
        this.plotData = this.edgeData.format(this._threshold);
        this.createScales();
        this.updateData();
        return this;
    }

    /**
     * Access / modify the height in a D3 style way. If modified, the component will update just the part that is needed to be updated
     */
    height(): number
    height(value: number): this
    height(value?) {
        if (value == null) {
            return this.options.height
        }

        this.options.height = value
        this.updateHeight()
        return this;
    }

    /**
     * Access / modify the width in a D3 style way. If modified, the component will update just the part that is needed to be updated
     */
    width(): number
    width(value: number): this
    width(value?: number): this | number {
        if (value == null) {
            return this.options.width;
        }
        this.options.width = value;
        this.updateWidth();
        return this;
    }

    /**
     * Access / modify the threshold in a D3 style way. If modified, the component will update just the part that is needed to be updated
     */
    threshold(): number
    threshold(value: number): this
    threshold(value?) {
        if (value == null) {
            return this._threshold;
        }

        this._threshold = value;
        this.plotData = this.edgeData.format(this._threshold);
        this.createScales();
        this.updateData();
        return this;
    }

    _wrangle(data: AttentionData) {
        return data;
    }

    _render(data: AttentionData) {
        this.svg.html('')
        this.updateHeight();
        this.updateWidth();

        this.updateData();
        return this;
    }
}