mirror of
https://github.com/PeWu/topola-viewer.git
synced 2025-12-23 18:50:04 +00:00
Added zooming capability
This feature is not enabled by default yet and can be only enabled by adding &enabledZoom=true to the URL
This commit is contained in:
parent
ac9f6f475d
commit
67c78b5982
@ -178,6 +178,7 @@ interface Arguments {
|
||||
chartType: ChartType;
|
||||
gedcom?: string;
|
||||
images?: Map<string, string>;
|
||||
enableZoom: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -224,6 +225,7 @@ function getArguments(location: H.Location<any>): Arguments {
|
||||
|
||||
gedcom: location.state && location.state.data,
|
||||
images: location.state && location.state.images,
|
||||
enableZoom: getParam('enableZoom') === 'true', // False by default.
|
||||
};
|
||||
}
|
||||
|
||||
@ -263,6 +265,8 @@ interface State {
|
||||
/** Source of the data. */
|
||||
source?: DataSourceEnum;
|
||||
loadingMore?: boolean;
|
||||
/** Whether the zoom functionality is enabled. */
|
||||
enableZoom: boolean;
|
||||
}
|
||||
|
||||
export class App extends React.Component<RouteComponentProps, {}> {
|
||||
@ -272,6 +276,7 @@ export class App extends React.Component<RouteComponentProps, {}> {
|
||||
standalone: true,
|
||||
chartType: ChartType.Hourglass,
|
||||
showErrorPopup: false,
|
||||
enableZoom: false,
|
||||
};
|
||||
chartRef: Chart | null = null;
|
||||
|
||||
@ -387,6 +392,7 @@ export class App extends React.Component<RouteComponentProps, {}> {
|
||||
standalone: args.standalone,
|
||||
chartType: args.chartType,
|
||||
source: args.source,
|
||||
enableZoom: args.enableZoom,
|
||||
}),
|
||||
);
|
||||
try {
|
||||
@ -405,6 +411,7 @@ export class App extends React.Component<RouteComponentProps, {}> {
|
||||
standalone: args.standalone,
|
||||
chartType: args.chartType,
|
||||
source: args.source,
|
||||
enableZoom: args.enableZoom,
|
||||
}),
|
||||
);
|
||||
} catch (error) {
|
||||
@ -438,6 +445,7 @@ export class App extends React.Component<RouteComponentProps, {}> {
|
||||
standalone: args.standalone,
|
||||
chartType: args.chartType,
|
||||
source: args.source,
|
||||
enableZoom: args.enableZoom,
|
||||
loadingMore: false,
|
||||
}),
|
||||
);
|
||||
@ -541,6 +549,7 @@ export class App extends React.Component<RouteComponentProps, {}> {
|
||||
chartType={this.state.chartType}
|
||||
onSelection={this.onSelection}
|
||||
ref={(ref) => (this.chartRef = ref)}
|
||||
enableZoom={this.state.enableZoom}
|
||||
/>
|
||||
{this.state.showSidePanel ? (
|
||||
<Responsive minWidth={768} id="sidePanel">
|
||||
|
||||
@ -15,21 +15,39 @@ import {
|
||||
CircleRenderer,
|
||||
} from 'topola';
|
||||
|
||||
/** Called when the view is dragged with the mouse. */
|
||||
function zoomed() {
|
||||
/**
|
||||
* Called when the view is dragged with the mouse.
|
||||
*
|
||||
* @param size the size of the chart
|
||||
*/
|
||||
function zoomed(size: [number, number], enableZoom: boolean) {
|
||||
const svg = d3.select('#chart');
|
||||
const parent = (svg.node() as HTMLElement).parentElement!;
|
||||
|
||||
if (enableZoom) {
|
||||
const scale = d3.event.transform.k;
|
||||
const offsetX = d3.max([0, (parent.clientWidth / scale - size[0]) / 2])!;
|
||||
const offsetY = d3.max([0, (parent.clientHeight / scale - size[1]) / 2])!;
|
||||
svg.attr(
|
||||
'transform',
|
||||
`translate(${-size[0] / 2}, ${-size[1] / 2})
|
||||
scale(${scale})
|
||||
translate(${size[0] / 2 + offsetX}, ${size[1] / 2 + offsetY})`,
|
||||
);
|
||||
}
|
||||
|
||||
parent.scrollLeft = -d3.event.transform.x;
|
||||
parent.scrollTop = -d3.event.transform.y;
|
||||
}
|
||||
|
||||
/** Called when the scrollbars are used. */
|
||||
function scrolled() {
|
||||
function scrolled(enableZoom: boolean) {
|
||||
const svg = d3.select('#chart');
|
||||
const parent = (svg.node() as HTMLElement).parentElement as Element;
|
||||
const x = parent.scrollLeft + parent.clientWidth / 2;
|
||||
const y = parent.scrollTop + parent.clientHeight / 2;
|
||||
d3.select(parent).call(d3.zoom().translateTo, x, y);
|
||||
const scale = enableZoom ? d3.zoomTransform(parent).k : 1;
|
||||
d3.select(parent).call(d3.zoom().translateTo, x / scale, y / scale);
|
||||
}
|
||||
|
||||
/** Loads blob as data URL. */
|
||||
@ -116,6 +134,7 @@ export interface ChartProps {
|
||||
selection: IndiInfo;
|
||||
chartType: ChartType;
|
||||
onSelection: (indiInfo: IndiInfo) => void;
|
||||
enableZoom: boolean;
|
||||
}
|
||||
|
||||
/** Component showing the genealogy chart and handling transition animations. */
|
||||
@ -185,13 +204,13 @@ export class Chart extends React.PureComponent<ChartProps, {}> {
|
||||
const parent = (svg.node() as HTMLElement).parentElement as Element;
|
||||
|
||||
d3.select(parent)
|
||||
.on('scroll', scrolled)
|
||||
.on('scroll', () => scrolled(this.props.enableZoom))
|
||||
.call(
|
||||
d3
|
||||
.zoom()
|
||||
.scaleExtent([1, 1])
|
||||
.scaleExtent(this.props.enableZoom ? [0.1, 2] : [1, 1])
|
||||
.translateExtent([[0, 0], chartInfo.size])
|
||||
.on('zoom', zoomed),
|
||||
.on('zoom', () => zoomed(chartInfo.size, this.props.enableZoom)),
|
||||
);
|
||||
|
||||
const scrollTopTween = (scrollTop: number) => {
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user