0

我正在尝试建立一个 React 项目,我想在其中使用 ol-ext 提供的一些工具。我环顾四周,从库的创建者那里找到了一些 Codesandbox 项目,但由于在项目中导入库的方式导致错误,我无法使它们正常工作。

ol-ext 和其他 Open Layers 版本的语法或不兼容是否存在问题?

4

1 回答 1

1

对于遇到同样问题的任何人,我都设法让它工作,并实现了交互转换功能示例。下面是代码:

JS:

// Import stylesheets
import './style.css';
import "ol/ol.css";
import "ol-ext/dist/ol-ext.css";

import Transform from "ol-ext/interaction/Transform";
import Stamen from 'ol/source/Stamen';
import { Map, View } from "ol";
import { defaults } from "ol/control";
import * as olEvents from 'ol/events';
import TileLayer from 'ol/layer/Tile';
import { Style, Fill, Text, Icon, Stroke, RegularShape } from "ol/style";
import {Polygon, LineString, Point, Circle} from 'ol/geom';
import VectorLayer from "ol/layer/Vector";
import VectorSource from "ol/source/Vector";
import Feature from "ol/Feature";


 var interaction = new Transform ({
      enableRotatedTransform: false,
      /* Limit interaction inside bbox * /
      condition: function(e, features) {
        return ol.extent.containsXY([-465960, 5536486, 1001630, 6514880], e.coordinate[0], e.coordinate[1]);
      },
      /* */
      addCondition: olEvents.condition,
      // filter: function(f,l) { return f.getGeometry().getType()==='Polygon'; },
      // layers: [vector],
      hitTolerance: 2,
       translateFeature: false,
       scale: true,
       rotate: true,
       translate: true,
       stretch: true
    });

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' })
            })
          }));



    // Layers
    var layers = [
      new TileLayer({ 
        title:'terrain-background',
        source: new Stamen({ layer: 'terrain' })
      })
    ]
    // The map
    var map = new Map({
      target: null,
        view: new View({
          zoom: 5,
          center: [261720, 5951081]
        }),
        controls: defaults({ "attribution": false }),
        layers: layers
      });

      // Style
    function getStyle(feature) {
      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: 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({
      name: 'Vecteur',
      source: new VectorSource({ wrapX: false }),
      style: getStyle
    })
    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';

 map.addInteraction(interaction);


class App extends Component {
    state = {
      name: 'React',
      styles: false,
      scale: true,
      stretch: true,
      disableStretch: false,
      rotate: true,
      translate: true,
      translateFeature: false,
      forceMatching: false,
      info: "Hello there"
    };


  setHandleStyleInput = (event) => {
      this.setState({styles: event.target.checked})
    if (!interaction instanceof Transform) return;
      if (event.target.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:'yellow'})
            }),
            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:'yellow' })
            }),
          }));
        // 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:'yellow' })
            })
          }));
          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:'yellow' })
          })
        }));
        interaction.style.scaleh3 = interaction.style.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:'yellow' })
            })
          }));
        interaction.style.scalev2 = interaction.style.scalev;
      } else {
        interaction.setDefaultStyle();
      }
      // Refresh
      interaction.set('translate', interaction.get('translate'));


  }

    setPropertieScale = (p) =>{

      this.setState({scale: p.target.checked})
      if (p.target.checked) this.setState({ disableStretch: false});
      else this.setState({ disableStretch: true});
    }

    setPropertieStretch = (p) =>{
      this.setState({stretch: p.target.checked})
    }

    setPropertieRotate = (p) =>{
      this.setState({rotate: p.target.checked})
    }

    setPropertieTranslate = (p) =>{
      this.setState({translate: p.target.checked})
    }
    setPropertieTranslateFeature = (p) =>{
      this.setState({translateFeature: p.target.checked})
    }

    componentDidMount() {


    map.setTarget("map");
    }

  componentDidUpdate(prevState) {
    /** Style the transform handles for the current interaction
    */
    if(this.state.scale !== prevState.scale || this.state.stretch !== prevState.stretch || this.state.rotate !== prevState.rotate || this.state.translate !== prevState.translate || this.state.styles !== prevState.styles){


    var interaction = new Transform ({
      enableRotatedTransform: false,
      /* Limit interaction inside bbox * /
      condition: function(e, features) {
        return ol.extent.containsXY([-465960, 5536486, 1001630, 6514880], e.coordinate[0], e.coordinate[1]);
      },
      /* */
      addCondition: olEvents.condition,
      // filter: function(f,l) { return f.getGeometry().getType()==='Polygon'; },
      // layers: [vector],
      hitTolerance: 2,
       translateFeature: this.state.translateFeature,
       scale: this.state.scale,
       rotate: this.state.rotate,
       translate: this.state.translate,
       stretch: this.state.stretch
    });
    interaction.set('translate', interaction.get('translate'));
    // Style handles

    }
  }

  render() {

    var info = "Heyyou"

     // Events handlers
    var startangle = 0;
    var d=[0,0];

    // Handle rotate on first point
    var firstPoint = false;
    interaction.on (['select'], function(e) {
      if (firstPoint && e.features && e.features.getLength()) {
        interaction.setCenter(e.features.getArray()[0].getGeometry().getFirstCoordinate());
      }
    });

    interaction.on (['rotatestart','translatestart'], function(e){
      // Rotation
      startangle = e.feature.get('angle')||0;
      // Translation
      d=[0,0];
    });
    interaction.on('rotating', (e)=>{
      info= "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', (e) =>{

      d[0]+=e.delta[0];
      d[1]+=e.delta[1];
      info= "translate: "+d[0].toFixed(2)+","+d[1].toFixed(2)


      if (firstPoint) {
        interaction.setCenter(e.features.getArray()[0].getGeometry().getFirstCoordinate());
      }
    });
    interaction.on('scaling', (e) => {
      info= "scale: "+e.scale[0].toFixed(2)+","+e.scale[1].toFixed(2)

      if (firstPoint) {
        interaction.setCenter(e.features.getArray()[0].getGeometry().getFirstCoordinate());
      }
    });
    interaction.on(['rotateend', 'translateend', 'scaleend'], (e) =>{
      this.setState({info: info})
      info= ""

    });

    return (
      <div>
        <div id="map" style={{width:"500px", height:"400px"}}></div>

  <div className="options" >
    <h2>Options:</h2>
    <ul><li>

      <input
            className="style"
            type="checkbox"
            checked={this.state.styles}
            onChange={this.setHandleStyleInput} />
      <label> styles transform handles (using fontawesome)</label>
    </li><li>
    <input
            className="scale"
            type="checkbox"
            checked={this.state.scale}
            onChange={this.setPropertieScale} /><label> enable scale</label>
    </li><li>
    <input
            className="stretch"
            type="checkbox"
            checked={this.state.stretch}
            onChange={this.setPropertieStretch} disabled={this.state.disableStretch} />
      <label> enable stretch</label>
    </li><li>
    <input
            className="rotate"
            type="checkbox"
            checked={this.state.rotate}
            onChange={this.setPropertieRotate} />
      <label> enable rotate</label>
    </li><li>
      <input
            className="translate"
            type="checkbox"
            checked={this.state.translate}
            onChange={this.setPropertieTranslate} />
      <label> enable translate</label>
    </li><li>
    <input
            className="translateFeature"
            type="checkbox"
            checked={this.state.translateFeature}
            onChange={this.setPropertieTranslateFeature} />
      <label> translate when click on feature</label>
    </li><li>
      SetRotateCenter:
      <button onClick={()=>{firstPoint=false; interaction.setCenter()}}>objects</button>
      <button onClick={()=>{firstPoint=false; interaction.setCenter(map.getView().getCenter())}}>view center</button>
      <button onClick={()=>{firstPoint=true;}}>first point</button>
    </li><li>
      <hr/>
      Use <i>Shift</i> to add object to tranform
      <hr/>
      Use <i>Shift</i> key to preserve proportions when scaling (see keepAspectRatio).
      <br />
      Use <i>Ctrl</i> key to modify the center when scaling.
    </li></ul>
    <div style={{background:"white", padding:"0 0.45em"}}><span id="info"></span>{this.state.info}</div>
  </div>
      </div>
    );
  }
}

HTML:

<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/4.7.0/css/font-awesome.min.css">

<div id="root"></div>

CSS:

h1, p {
  font-family: Lato;
}

a.icss-github-corner,
a.icss-github-corner-left {
  font-size: 2.5em;
  position: fixed;
  top:0;
  right: 0;
  z-index: 1000;
  color: #fff;
  background-color: #333;
  padding: .5em 2em 0;
  text-align: center;
  -webkit-transform: rotate(45deg);
      -ms-transform: rotate(45deg);
          transform: rotate(45deg);
  -webkit-transform-origin: 3.9em 3em;
      -ms-transform-origin: 3.9em 3em;
          transform-origin: 3.9em 3em;
  overflow: hidden;
}
a.icss-github-corner-left {
  left: 0;
  right: auto;
  -webkit-transform: rotate(-45deg);
      -ms-transform: rotate(-45deg);
          transform: rotate(-45deg);
  -webkit-transform-origin: 1em 3.1em;
      -ms-transform-origin: 1em 3.1em;
          transform-origin: 1em 3.1em;
}
a.icss-github-corner:hover i,
a.icss-github-corner-left:hover i{
  -webkit-animation: vertical 2s ease;
          animation: vertical 2s ease;
}

a.icss-github-corner i,
a.icss-github-corner-left i,
i.icss-github-corner {
  color: #fff;
  position: relative;
  display:inline-block;
  font-style: normal;
  background-color:currentColor;
  -webkit-box-sizing: border-box;
          box-sizing: border-box;
  vertical-align: middle;
  width: .8em;
  height: .6em;
  -webkit-border-radius: 45% 45% 45% 45% / 50%;
          border-radius: 45% 45% 45% 45% / 50%;
  background-color: currentColor;
  -webkit-box-shadow: 0 .35em 0 -.2em, 
      0 .38em 0 -.2em,
      0 .41em 0 -.2em,
      0 .44em 0 -.2em,
      0 .47em 0 -.2em;
          box-shadow: 0 .35em 0 -.2em, 
      0 .38em 0 -.2em,
      0 .41em 0 -.2em,
      0 .44em 0 -.2em,
      0 .47em 0 -.2em;
  margin: .12em .1em .23em;
}
a.icss-github-corner i:before,
a.icss-github-corner-left i:before,
i.icss-github-corner:before {
  content: "";
  border-width: 0; 
  position: absolute;
  -webkit-box-sizing: border-box;
          box-sizing: border-box;
  border-width: .15em .15em;
  border-style: solid;
  -webkit-border-radius: 0.02em 60% 100% 80%;
          border-radius: 0.02em 60% 100% 80%;
  left: 0;
  top: -.07em;
  -webkit-transform: rotate(20deg);
      -ms-transform: rotate(20deg);
          transform: rotate(20deg);
}
a.icss-github-corner i:after,
a.icss-github-corner-left i:after,
i.icss-github-corner:after {
  content: "";
  border-width: 0; 
  position: absolute;
  -webkit-box-sizing: border-box;
          box-sizing: border-box;
  border-width: .15em .15em;
  border-style: solid;
  -webkit-border-radius: 0.02em 80% 100% 60%;
          border-radius: 0.02em 80% 100% 60%;
  left: .5em;
  top: -.07em;
  -webkit-transform: rotate(65deg);
      -ms-transform: rotate(65deg);
          transform: rotate(65deg);
}
@-webkit-keyframes vertical {
    0%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    4%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    8%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    12%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    16%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    20%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    22%,100%{-webkit-transform:translate(0,0);transform:translate(0,0)}
}
@keyframes vertical {
    0%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    4%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    8%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    12%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    16%{-webkit-transform:translate(0,-3px);transform:translate(0,-3px)}
    20%{-webkit-transform:translate(0,3px);transform:translate(0,3px)}
    22%,100%{-webkit-transform:translate(0,0);transform:translate(0,0)}
}
/**/
body {
  font-family: 'Lucida Grande',Verdana,Geneva,Lucida,Arial,Helvetica,sans-serif;
    font-size: 16px;
}
a, i, b {
  color: #337ab7;
    text-decoration: none;
}
button i {
  color: #fff;
}
.ol-control.ol-bar .ol-control button i {
  color: #fff;
}
a:hover {
  text-decoration: underline;
}
a.title {
  text-decoration: none;
}
h1 {
  background: #1f6b75 url("") no-repeat scroll 10px center;
    color: #fff;
    font-size: 1.5em;
    padding: 0.5em 50px;
    margin:0;
}
h2 {
  color: #337ab7;
    font-size:1.1em;
    margin: 0.5em 0;
}

.info {
  background:#f5f5f5;
    padding:0.5em;
    margin: 1em 0;
}
.info ul {
  margin:0;
}

#map {
  float:left;
    margin-right:1em;
    background:#ddd;
}

.ol-attribution img {
  vertical-align:middle;
}

.layerSwitcher {
  display:inline-block;
    background:#cdf;
    padding:0.5em;
}
.btn {
  color:#fff;
    background:#369;
    padding:0.5em;
    text-decoration:none;
    cursor:pointer;
    display:inline-block;
    margin-right:0.5em;
}

.block,
.options {
  display: table;
    margin: 0.5em;
    position: relative;
    z-index: 1;
    margin:1em;
}
.options {
  background: #def;
    padding: 0.5em;
}
.options ul {
  list-style: none;
    padding-left: 0.5em;
}

i[class*="icss-"] {
  position: relative;
  display:inline-block;
  font-style: normal;
  background-color:currentColor;
  box-sizing: border-box;
  vertical-align: middle;
  font-size: 1.5em;
}
i[class*="icss-"]:before, 
i[class*="icss-"]:after {
  content: "";
  border-width: 0; 
  position: absolute;
}
i.icss-book {
  width:1em;
  height:.8em;
  background-color: transparent; 
  margin: 0 .03em .08em 0;
}
i.icss-book:before {
  height: .8em;
  width: 0.7em;
  box-shadow: inset 0 0 0 0.15em, 
      inset 0 -.48em,
      .07em .07em;
  border: 0.07em solid transparent;
  border-width: 0 0.07em .07em 0;
  border-radius: .05em .15em .1em .1em / .05em .05em .1em .05em;
  transform: skewX(-20deg);
  left: 0.15em;
}
i.icss-book:after {
  width: .2em;
  height: .2em;
  background-color: transparent;
  border: 0.05em solid currentColor;
  border-color: currentColor transparent transparent currentColor;
  border-radius: 50%;
  transform: rotate(-45deg);
  top: 0.67em;
  left: .018em;
  box-shadow: .13em -.15em 0 -.05em,
      .51em -.33em 0 -.05em;
}
.experimental {
  color:#fff; 
  background: #f91; 
  padding:.2em .5em;
  display: inline-block;
  -webkit-transform: rotate(-5deg);
  transform: rotate(-5deg);
  margin: -1em 0;
}

.ol-attribution ul {
  font-size: .8em;
}

这里还有一个工作示例:StackBlitz

于 2020-06-16T13:55:36.207 回答