Security News
pnpm 10.0.0 Blocks Lifecycle Scripts by Default
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
@asymmetrik/ngx-leaflet-draw
Advanced tools
Leaflet Draw extension to the @asymmetrik/ngx-leaflet package for Angular.io Provides Leaflet Draw integration into Angular.io projects. Compatible with Leaflet v1.x and Leaflet Draw 1.x
Now supports Angular v18, and use in Angular-CLI based projects Given the lack of activity in the Leaflet.draw project, we are unlikely to extend this plugin with additional functionality beyond Angular upgrades.
NOTE: This is the last version of this library that will be published under the namespace @asymmetrik. This and future packages will be released under the @bluehalo namespace.
Install the package and its peer dependencies via npm (or yarn):
npm install leaflet
npm install leaflet-draw
npm install @asymmetrik/ngx-leaflet
npm install @asymmetrik/ngx-leaflet-draw
NOTE: I've seen some issues with leaflet-draw@1.0.3
and 1.0.4
.
Specifically, label anchors and drawing rectangles doesn't seem to work correctly.
If you have issues, leaflet-draw@1.0.2
seems to be the latest release without those issues.
If you intend to use this library in a typescript project (utilizing the typings), you will need to also install the leaflet typings via npm:
npm install --save-dev @types/leaflet
npm install --save-dev @types/leaflet-draw
If you want to run the demo, clone the repository, perform an npm install
, npm run demo
and then go to http://localhost:4200.
To use this library, there are a handful of setup steps to go through that vary based on your app environment (e.g., Webpack, ngCli, SystemJS, etc.). Generally, the steps are:
For leaflet to work, you need to have the leaflet stylesheets loaded into your application.
If you've installed via npm, you will need to load ./node_modules/leaflet/dist/leaflet.css
and ./node_modules/leaflet-draw/dist/leaflet.draw.css
.
How you include the stylesheet will depend on your specific setup. For examples, refer to the @asymmetrik/ngx-leaflet README
This project is exported using UMD and it includes typings. So, you shouldn't have to do anything special to use it if you're building your project in Typescript.
Before you can use the module in your Angular.io app, you'll need to import it in your application. Note that you also need to import the ngx-leaflet module as well.
For example, in your app.module.ts
, add:
import { LeafletModule } from '@asymmetrik/ngx-leaflet';
import { LeafletDrawModule } from '@asymmetrik/ngx-leaflet-draw';
...
imports: [
...
LeafletModule,
LeafletDrawModule
]
...
You brave soul. The code is exported using UMD (bundles are in the ./dist dir) so you should be able to import is using whatever module system/builder you're using, even if you aren't using Typescript.
To create a map, use the leaflet
attribute directive. This directive must appear first.
You must specify an initial zoom/center and set of layers either via leafletOptions
or by binding to leafletZoom
, leafletCenter
, and leafletLayers
.
Finally, add the leafletDraw
attribute directive to add the leaflet draw control and configure it with leafletDrawOptions
.
<div leaflet style="height: 400px;"
leafletDraw
[leafletOptions]="options"
[leafletDrawOptions]="drawOptions"
(leafletDrawCreated)="onDrawCreated($event)">
<div [leafletLayer]="drawnItems"></div>
</div>
To enable editing, you need to add a featureGroup
to the map and pass the feature group in with the drawOptions
.
In addition, you will need to add layers to the feature group yourself, as this will no longer happen automatically.
Both of these changes are new in @asymmetrik/ngx-leaflet-draw@6
, and were made to match default Leaflet Draw behavior.
drawnItems: FeatureGroup = featureGroup();
drawOptions = {
edit: {
featureGroup: this.drawnItems
}
};
public onDrawCreated(e: any) {
this.drawnItems.addLayer((e as DrawEvents.Created).layer);
}
This is an attribute directive that initiates the leaflet draw plugin.
Input binding for the options to be passed to the draw plugin upon creation. These options are only currently processed at creation time.
drawOptions = {
position: 'topright',
draw: {
marker: {
icon: L.icon({
iconSize: [ 25, 41 ],
iconAnchor: [ 13, 41 ],
iconUrl: '2b3e1faf89f94a4835397e7a43b4f77d.png',
iconRetinaUrl: '680f69f3c2e6b90c1812a813edf67fd7.png',
shadowUrl: 'a0c6cc1401c107b501efee6477816891.png'
})
},
polyline: false,
circle: {
shapeOptions: {
color: '#d4af37'
}
},
rectangle: {
shapeOptions: {
color: '#85bb65'
}
}
},
edit: {
featureGroup: this.drawnItems
}
};
The options object is passed through to the Leaflet.draw object. Therefore, you can reference their documentation for help configuring the draw control.
This object is copied into L.drawLocal
to set localization options.
These settings are only applied at creation time.
drawLocal: any = {
draw: {
toolbar: {
buttons: {
polygon: 'Draw an awesome polygon!'
}
}
}
};
This leaflet draw plugin exposes the L.Draw.Event
s found in the leaflet draw documentation.
For a working example, check out the the demo.
The following events are provided:
(leafletDrawCreated)
(leafletDrawEdited)
(leafletDrawDeleted)
(leafletDrawStart)
(leafletDrawStop)
(leafletDrawVertex)
(leafletDrawEditStart)
(leafletDrawEditMove)
(leafletDrawEditResize)
(leafletDrawEditVertex)
(leafletDrawEditStop)
(leafletDrawDeleteStart)
(leafletDrawDeleteStop)
(leafletDrawToolbarOpened)
(leafletDrawToolbarClosed)
(leafletDrawMarkerContext)
Occasionally, you may need to directly access the Leaflet Draw Control instance. For example, to dynamically change settings or change its behavior. There are a couple of different ways to achieve this depending on what you're trying to do.
The easiest and most flexible way is to use the output binding leafletDrawReady
.
This output is invoked after the map is created, the argument of the event being the Control.Draw
instance.
The second is to get a reference to the leaflet draw directive itself - and there are a couple of ways to do this.
With a reference to the directive, you can invoke the getDrawControl()
function to get a reference to the Control.Draw
instance.
This output is emitted when once when the Leaflet Draw Control is initially created inside of the Leaflet Draw directive. The event will only fire when the Control exists and is ready for manipulation.
<div leaflet
leafletDraw
[leafletOptions]="options"
[leafletDrawOptions]="drawOptions"
(leafletDrawReady)="onDrawReady($event)">
</div>
onDrawReady(drawControl: Draw.Control) {
// Do stuff with map
}
This method of getting the draw control makes the most sense if you are using the Leaflet directive inside your own component and just need to add some limited functionality or register some event handlers.
In Angular.io, directives are injectable the same way that Services are.
This means that you can create your own component or directive and inject the LeafletDrawDirective
into it.
This will only work if your custom component/directive exists on the same DOM element and is ordered after the injected LeafletDrawDirective, or if it is on a child DOM element.
<!-- On the same DOM element -->
<div leaflet leafletDraw myCustomDirective>
</div>
<!-- On a child DOM element -->
<div leaflet leafletDraw>
<div myCustomDirective></div>
</div>
@Directive({
selector: '[myCustomDirective]'
})
export class MyCustomDirective {
leafletDrawDirective: LeafletDrawDirective;
constructor(leafletDrawDirective: LeafletDrawDirective) {
this.leafletDrawDirective = leafletDrawDirective;
}
someFunction() {
if (null != this.leafletDrawDirective.getDrawControl()) {
// Do stuff with the draw control
}
}
}
The benefit of this approach is it's a bit cleaner if you're interested in adding some reusable capability to the existing leaflet draw directive.
If you want to toggle the draw control on and off, you can use the following approach:
<button (click)="shown = !shown">Show/Hide Control</button>
<div leaflet style="height: 400px;"
[leafletOptions]="options">
<div *ngIf="shown"
leafletDraw
[leafletDrawOptions]="drawOptions"></div>
</div>
You can place the leafletDraw directive on a child element and then use *ngIf to add/remove the draw control from the map. When ngIf evaluates to false, the child element is removed from the map, which destroys the control. When it evaluates to true, the child element is added to the map, which recreates the control.
If you are using Angular CLI or Webpack to package your project, you will need to configure the marker icon as shown in the leafletDrawOptions
example above.
The issue has to do with how Leaflet handles icon image loading.
For more details on how to set this up, reference the README from @asymmetrik/ngx-leaflet.
PRs accepted. If you are part of Asymmetrik, please make contributions on feature branches off of the develop
branch.
If you are outside of Asymmetrik, please fork our repo to make contributions and submit PRs against develop
.
See LICENSE in repository for details.
Leaflet Is an awesome mapping package. Leaflet.draw Makes drawing shapes on maps super easy.
FAQs
Angular.io components for the Leaflet Draw plugin
We found that @asymmetrik/ngx-leaflet-draw demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 7 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
Product
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
Research
Security News
Socket researchers have discovered multiple malicious npm packages targeting Solana private keys, abusing Gmail to exfiltrate the data and drain Solana wallets.