Skip to content

Commit

Permalink
Fix wrong type in transform (#99)
Browse files Browse the repository at this point in the history
Fixes #98

Add example
  • Loading branch information
Siedlerchr authored Sep 28, 2022
1 parent a538e9c commit dbed8b5
Show file tree
Hide file tree
Showing 3 changed files with 279 additions and 5 deletions.
39 changes: 35 additions & 4 deletions @types/ol-ext/interaction/Transform.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,8 @@ type TransformOnSignature<Return> = OnSignature<EventTypes, Event, Return> &
OnSignature<Types | 'rotatestart' | 'rotating' | 'rotateend', RotateEvent, Return> &
OnSignature<Types | 'scalestart' | 'scaling' | 'scaleend', ScaleEvent, Return> &
OnSignature<Types | 'translatestart' | 'translating' | 'translateend', TranslateEvent, Return> &
CombinedOnSignature<Types | EventTypes | 'change' | 'change:active' | 'error' | 'propertychange' | 'rotatestart' | 'rotating' | 'rotateend' | 'scalestart' | 'scaling' | 'scaleend' | 'translatestart' | 'translating' | 'translateend', Return>
OnSignature<Types | 'select' , SelectEvent, Return> &
CombinedOnSignature<Types | EventTypes | 'change' | 'change:active' | 'error' | 'propertychange' | 'rotatestart' | 'rotating' | 'rotateend' | 'scalestart' | 'scaling' | 'scaleend' | 'translatestart' | 'translating' | 'translateend' | 'select', Return>

export interface Options {
filter?: (f: Feature, l: Layer) => boolean;
Expand Down Expand Up @@ -59,6 +60,7 @@ export enum TranslateEventType {
TRANSLATEEND = 'translateend',
}

export type SelectEventType = 'select';
/**
* Interaction transform
*
Expand All @@ -85,13 +87,29 @@ export enum TranslateEventType {
* @param {number|Array<number>|function} [options.pointRadius=0] radius for points or a function that takes a feature and returns the radius (or [radiusX, radiusY]). If not null show handles to transform the points
*/

export type olExtStyle = 'default' | 'translate' | 'rotate' | 'rotate0' | 'scale' | 'scale1' | 'scale2' | 'scale3' | 'scalev' | 'scaleh1' | 'scalev2' | 'scaleh3'
export default class Transform extends Pointer {
constructor(options?: Options);
/**
* Cursors for transform
*/
Cursors: any;
/**
Cursors: {
default: string;
select: string;
translate: string;
rotate: string;
rotate0: string;
scale: string;
scale1: string;
scale2: string;
scale3: string;
scalev: string;
scaleh1: string;
scalev2: string;
scaleh3: string;
};
/**
* Remove the interaction from its current map, if any, and attach it to a new
* map, if any. Pass `null` to just remove the interaction from the current map.
* @param map Map.
Expand All @@ -118,7 +136,7 @@ export default class Transform extends Pointer {
* @param olstyle
* @api stable
*/
setStyle(style: Style, olstyle: Style | Style[]): void;
setStyle(style: olExtStyle, olstyle: Style | Style[]): void;
/**
* Draw transform sketch
* @param draw only the center
Expand Down Expand Up @@ -212,3 +230,16 @@ export class TranslateEvent extends BaseEvent {
pixel: Pixel;
coordinate: Coordinate;
}

export class SelectEvent extends BaseEvent{
constructor(
type: SelectEventType,
feature: Feature<Geometry>,
features: Collection<Feature<Geometry>>,
)
feature: Feature<Geometry>;
features: Collection<Feature<Geometry>>;
pixel: Pixel;
coordinate: Coordinate;

}
2 changes: 1 addition & 1 deletion examples/interaction/map.interaction.clip.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,4 +33,4 @@ const map = new Map({
// Clip interaction
const clip = new Clip({ radius: Number($("#radius").val()), layers:stamen });
map.addInteraction(clip);
window.clip = clip;
window.clip = clip;
243 changes: 243 additions & 0 deletions examples/interaction/map.interaction.transform.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,243 @@
import { Map, View } from 'ol';
import { Tile } from 'ol/layer';
import { Stamen } from 'ol/source';
import Transform from 'ol-ext/interaction/Transform';
import Feature from 'ol/Feature';
import { Style, RegularShape, Fill, Stroke, Text } from 'ol/style';
import VectorLayer from 'ol/layer/Vector';
import VectorSource from 'ol/source/Vector';
import { Circle, LineString, Point, Polygon, Geometry } from 'ol/geom';
import { FeatureLike } from 'ol/Feature';
import { always, Condition, shiftKeyOnly } from 'ol/events/condition';
import { SelectEvent } from 'ol-ext/interaction/Transform';
import { Type } from 'ol/geom/Geometry';
import { KeyEvent } from 'ol-ext/interaction/CurrentMap';

declare global {
interface Window {
$(selector: any, context?: any): any;
// @ts-ignore
interaction: Transform
}
}
const $ = window.$;
// Layers
var layers = [
new Tile({
//title:'terrain-background',
source: new Stamen({ layer: 'terrain' })
})
]
// The map
var map = new Map({
target: 'map',
view: new View({
zoom: 5,
center: [261720, 5951081]
}),
layers: layers
});

// Style
export function getStyle(feature: FeatureLike, resolution: number) {
return [new Style({
image: new RegularShape({
fill: new Fill({ color: [0, 0, 255, 0.4] }),
stroke: new Stroke({ color: [0, 0, 255, 1], width: 1 }),
radius: feature.get('radius') || 10,
points: 3,
angle: feature.get('angle') || 0
}),
fill: new Fill({ color: [0, 0, 255, 0.4] }),
stroke: new Stroke({ color: [0, 0, 255, 1], width: 1 })
})];
}

// New vector layer
var vector = new VectorLayer({
source: new VectorSource({ wrapX: false }),
style: getStyle
})
vector.set("name", "Vecteur")
map.addLayer(vector);
vector.getSource()?.addFeature(new Feature(new Polygon([[[34243, 6305749], [-288626, 5757848], [210354, 5576845], [300000, 6000000], [34243, 6305749]]])));
vector.getSource()?.addFeature(new Feature(new LineString([[406033, 5664901], [689767, 5718712], [699551, 6149206], [425601, 6183449]])));
vector.getSource()?.addFeature(new Feature(new Point([269914, 6248592])));
vector.getSource()?.addFeature(new Feature(new Circle([500000, 6400000], 100000)));

// Set cursor style

Transform.prototype.Cursors['rotate'] = 'url(\'\') 5 5, auto';
Transform.prototype.Cursors['rotate0'] = 'url(\'\') 5 5, auto';

/** Style the transform handles for the current interaction
*/
function setHandleStyle() {
if (!interaction as any instanceof Transform) return;
if ($("#style").prop('checked')) {
// Style the rotate handle
var circle = new RegularShape({
fill: new Fill({ color: [255, 255, 255, 0.01] }),
stroke: new Stroke({ width: 1, color: [0, 0, 0, 0.01] }),
radius: 8,
points: 10
});
interaction.setStyle('rotate',
new Style({
text: new Text({
text: '\uf0e2',
font: "16px Fontawesome",
textAlign: "left",
fill: new Fill({ color: 'red' })
}),
image: circle
}));
// Center of rotation
interaction.setStyle('rotate0',
new Style({
text: new Text({
text: '\uf0e2',
font: "20px Fontawesome",
fill: new Fill({ color: [255, 255, 255, 0.8] }),
stroke: new Stroke({ width: 2, color: 'red' })
}),
}));
// Style the move handle
interaction.setStyle('translate',
new Style({
text: new Text({
text: '\uf047',
font: "20px Fontawesome",
fill: new Fill({ color: [255, 255, 255, 0.8] }),
stroke: new Stroke({ width: 2, color: 'red' })
})
}));
// Style the strech handles
/* uncomment to style * /
interaction.setStyle ('scaleh1',
new Style({
text: new Text ({
text:'\uf07d',
font:"bold 20px Fontawesome",
fill: new Fill({ color:[255,255,255,0.8] }),
stroke: new Stroke({ width:2, color:'red' })
})
}));
interaction.scaleh3 = interaction.scaleh1;
interaction.setStyle('scalev',
new Style({
text: new Text ({
text:'\uf07e',
font:"bold 20px Fontawesome",
fill: new Fill({ color:[255,255,255,0.8] }),
stroke: new Stroke({ width:2, color:'red' })
})
}));
interaction.scalev2 = interaction.scalev;
/**/
} else {
interaction.setDefaultStyle();
}
// Refresh
interaction.set('translate', interaction.get('translate'));
};

/** Set properties
*/
export function setPropertie(p: string) {
interaction.set(p, $("#" + p).prop('checked'));
if (!$("#scale").prop("checked")) $("#stretch").prop('disabled', true);
else $("#stretch").prop('disabled', false);
}

export function setAspectRatio(p: string) {
if ($("#" + p).prop('checked')) interaction.set("keepAspectRatio", always);
else interaction.set("keepAspectRatio", function (e: KeyEvent) { return e.originalEvent.shiftKey });
}

var interaction = new Transform({
enableRotatedTransform: false,
/* Limit interaction inside bbox * /
condition: function(e, features) {
return extent.containsXY([-465960, 5536486, 1001630, 6514880], e.coordinate[0], e.coordinate[1]);
},
/* */
addCondition: shiftKeyOnly,
// filter: function(f,l) { return f.getGeometry().getType()==='Polygon'; },
// layers: [vector],
hitTolerance: 2,
translateFeature: $("#translateFeature").prop('checked'),
scale: $("#scale").prop('checked'),
rotate: $("#rotate").prop('checked'),
keepAspectRatio: $("#keepAspectRatio").prop('checked') ? always : undefined,
keepRectangle: false,
translate: $("#translate").prop('checked'),
stretch: $("#stretch").prop('checked'),
// Get scale on points
pointRadius: function (f) {
var radius = f.get('radius') || 10;
return [radius, radius];
}
});
map.addInteraction(interaction);
// Style handles
setHandleStyle();
// Events handlers
var startangle = 0;
var startRadius = 10;
var d = [0, 0];


type GeoWithoutPoint = Exclude<Type, 'Point'>
// Handle rotate on first point
var firstPoint = false;
interaction.on('select', function (e: SelectEvent) {
if (firstPoint && e.features && e.features.getLength()) {

const geo = e.features.getArray()[0].getGeometry() as any;
interaction.setCenter(geo.getFirstCoordinate());
}
});
// ts does not have a XOR type
interaction.on(['rotatestart', 'translatestart', 'scalestart'], function (e: any) {
// Rotation
startangle = e.feature.get('angle') || 0;
// radius
startRadius = e.feature.get('radius') || 10;
// Translation
d = [0, 0];
});
interaction.on('rotating', function (e) {
$('#info').text("rotate: " + ((e.angle * 180 / Math.PI - 180) % 360 + 180).toFixed(2));
// Set angle attribute to be used on style !
e.feature.set('angle', startangle - e.angle);
});
interaction.on('translating', function (e) {
d[0] += e.delta[0];
d[1] += e.delta[1];
$('#info').text("translate: " + d[0].toFixed(2) + "," + d[1].toFixed(2));
if (firstPoint) {

const point = e.features.getArray()[0].getGeometry() as any
interaction.setCenter(point.getFirstCoordinate());
}
});
interaction.on('scaling', function (e) {
$('#info').text("scale: " + e.scale[0].toFixed(2) + "," + e.scale[1].toFixed(2));
if (firstPoint) {
interaction.setCenter((e.features.getArray()[0].getGeometry() as any).getFirstCoordinate());
}
if (e.features.getLength() === 1) {
var feature = e.features.item(0);
feature.set('radius', startRadius * Math.abs(e.scale[0]));
}
});
interaction.on(['rotateend', 'translateend', 'scaleend'], function (e) {
$('#info').text("");
});
// @ts-ignore
window.interaction = interaction;
window.setHandleStyle = setHandleStyle;
window.setPropertie = setPropertie;
window.setAspectRatio = setAspectRatio;

0 comments on commit dbed8b5

Please sign in to comment.