Adding a marker to the map

Open on CodeSandbox

<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1" />
    <script src="https://js.api.mappable.world/v3/?apikey=<YOUR_APIKEY>&lang=en_US" type="text/javascript"></script>
    <script src="./common.js"></script>

    <script>
      window.map = null;

      main();
      async function main() {
        await mappable.ready;
        const {MMap, MMapDefaultSchemeLayer, MMapMarker, MMapControls, MMapDefaultFeaturesLayer} = mappable;

        const {MMapZoomControl} = await mappable.import('@mappable-world/mappable-controls@0.0.1');
        const {MMapDefaultMarker} = await mappable.import('@mappable-world/mappable-markers@0.0.1');

        map = new MMap(document.getElementById('app'), {location: LOCATION});

        map.addChild((scheme = new MMapDefaultSchemeLayer()));
        map.addChild(new MMapControls({position: 'right'}).addChild(new MMapZoomControl({})));
        map.addChild(new MMapDefaultFeaturesLayer({id: 'features'}));

        POINTS.forEach((point) => {
          if (point.element) {
            map.addChild(new MMapMarker(point, point.element(point)));
          } else {
            map.addChild(new MMapDefaultMarker(point));
          }
        });

        const marker = new MMapDefaultMarker(INC_POINT);
        map.addChild(marker);

        const marker2 = new MMapDefaultMarker(INC2_POINT);
        map.addChild(marker2);

        let inc = 0;
        const updateTitle = () => {
          inc++;
          marker.update({
            title: 'Marker inc #' + inc
          });
        };

        updateTitle();
        setInterval(updateTitle, 1000);

        setTimeout(() => {
          marker2.update({
            title: 'Marker 2',
            subtitle: 'Marker 2'
          });
        }, 1000);
      }
    </script>

    <!-- prettier-ignore -->
    <style> html, body, #app { width: 100%; height: 100%; margin: 0; padding: 0; font-family: Arial, Helvetica, sans-serif; } .toolbar { position: absolute; z-index: 1000; top: 0; left: 0; display: flex; align-items: center; padding: 16px; } .toolbar a { padding: 16px; }  </style>
    <link rel="stylesheet" href="./common.css" />
  </head>
  <body>
    <div id="app"></div>
  </body>
</html>
.pie-marker {
  position: relative;

  overflow: hidden;

  width: 50px;
  height: 50px;

  border-radius: 50%;
  background-color: currentColor;
}

.pie-marker-title {
  position: absolute;
  top: 120%;
  left: 50%;

  padding: 2px 4px;

  background-color: #fff;

  transform: translateX(-50%);
}

:root {
  --radius: 20px;
  --icon: #fff;
}

.circle {
  position: relative;

  width: var(--radius, 20px);
  height: var(--radius, 20px);

  border-radius: 50%;
  background-color: currentColor;
}

.circle::before {
  position: absolute;
  top: 50%;
  left: 50%;

  display: inline-block;

  width: 50%;
  height: 50%;

  content: '';

  border-radius: 50%;
  background: no-repeat center center #fff;
  background-image: var(--icon);

  transform: translate3d(-50%, -50%, 0);
}

:root {
  --size: 50px;
}

.icon {
  position: relative;

  width: var(--size, 20px);
  height: var(--size, 20px);

  background: no-repeat center center;
  background-size: contain;
}

.icon-title {
  position: absolute;
  top: 120%;
  left: 50%;

  padding: 2px 4px;

  background-color: #fff;

  transform: translateX(-50%);
}
const LOCATION = {center: [55.45971, 25.25359], zoom: 10};

const INC_POINT = {coordinates: [55.76971, 25.39359], title: 'Marker inc #0'};
const INC2_POINT = {coordinates: [55.91971, 25.29359], color: '#fcc', draggable: true};

const POINTS = [
  {coordinates: [55.61971, 25.29359]},
  {
    coordinates: [55.41971, 25.34059],
    title: 'Diagram',
    color: '#0E4779',
    draggable: true,
    colors: [
      {percentage: 30, color: '#0E4779'},
      {percentage: 20, color: '#1E98FF'},
      {percentage: 40, color: '#82CDFF'},
      {percentage: 10, color: '#ff9f82'}
    ],
    element: diagram
  },
  {
    coordinates: [55.55823, 25.17835],
    color: '#0095b6',
    title: 'color <strong>bondi beach water<strong>',
    draggable: true
  },
  {
    coordinates: [55.53489, 25.32702],
    color: '#735184',
    title: '<strong>Silver crimson<strong> color',
    draggable: true
  },
  {
    coordinates: [55.3495, 25.18068],
    color: '#3caa3c',
    title: 'love toad color',
    draggable: true,
    element: circle
  },
  {
    coordinates: [55.76971, 25.27598],
    color: 'yellow',
    title: 'color <strong>sun<strong>',
    draggable: true,
    onClick: () => alert('click')
  },
  {
    coordinates: [55.47584, 25.13565],
    title: 'color <strong>red<strong>',
    size: '60px',
    icon: '',
    draggable: true,
    element: icon,
    onDoubleClick: () => alert('Double click')
  },
  {
    coordinates: [55.30692, 25.32007],
    title: 'the color of <strong>Pacific Ocean<strong>',
    color: '#3b5998',
    draggable: true,
    mapFollowsOnDrag: true
  },
  {
    coordinates: [55.25474, 25.18843],
    color: '#477510',
    title: 'nose color Donatello',
    subtitle: 'Very long but incredibly interesting text',
    draggable: true
  },
  {
    coordinates: [55.35474, 25.09359],
    color: '#343d44',
    title: 'Hello!',
    subtitle: 'Very long but <br>incredibly interesting text',
    draggable: true
  },
  {
    coordinates: [55.63376, 25.28373],
    title: 'blue color',
    icon: 'url()',
    color: '#51aabd',
    radius: '50px',
    draggable: true,
    element: circle,
    onFastClick: () => alert('Fast click')
  }
];

function diagram(props) {
  const div = document.createElement('div');

  const diagram = document.createElement('div');

  diagram.className = 'pie-marker';
  diagram.style.color = props.color;

  const gradient = [];
  let previous = 0;
  for (let i = 0; i < props.colors.length; i += 1) {
    const p = props.colors[i];
    const deg = (360 / 100) * p.percentage;
    gradient.push(`${p.color} ${previous}deg ${previous + deg}deg`);
    previous = previous + deg;
  }

  diagram.style.background = 'conic-gradient(' + gradient.join(', ') + ')';

  const title = document.createElement('div');
  title.innerHTML = props.title;
  title.className = 'pie-marker-title';

  div.appendChild(diagram);
  div.appendChild(title);

  return div;
}

function circle(props) {
  const circle = document.createElement('div');
  circle.classList.add('circle');
  circle.style.color = props.color;
  props.radius && circle.style.setProperty('--radius', props.radius);
  props.icon && circle.style.setProperty('--icon', props.icon);
  circle.title = props.title;
  return circle;
}

function icon(props) {
  const circle = document.createElement('div');
  circle.classList.add('icon');
  circle.style.color = props.color;
  circle.style.backgroundImage = `url(${props.icon})`;
  circle.style.setProperty('--size', props.size);

  if (props.title) {
    const title = document.createElement('div');
    title.innerHTML = props.title;
    title.className = 'icon-title';
    circle.appendChild(title);
  }

  return circle;
}
<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1" />
    <script crossorigin src="https://unpkg.com/react@17/umd/react.production.min.js"></script>
    <script crossorigin src="https://unpkg.com/react-dom@17/umd/react-dom.production.min.js"></script>
    <script crossorigin src="https://unpkg.com/babel-standalone@6/babel.min.js"></script>
    <script src="https://js.api.mappable.world/v3/?apikey=<YOUR_APIKEY>&lang=en_US" type="text/javascript"></script>
    <script src="./common.js"></script>

    <script type="text/babel">
      window.map = null;

      main();
      async function main() {
        const [mappableReact] = await Promise.all([
          mappable.import('@mappable-world/mappable-reactify'),
          mappable.ready
        ]);
        const reactify = mappableReact.reactify.bindTo(React, ReactDOM);
        const {MMap, MMapDefaultSchemeLayer, MMapDefaultFeaturesLayer, MMapMarker, MMapControls} =
          reactify.module(mappable);

        const {MMapZoomControl} = reactify.module(await mappable.import('@mappable-world/mappable-controls@0.0.1'));
        const {MMapDefaultMarker} = reactify.module(await mappable.import('@mappable-world/mappable-markers@0.0.1'));

        function App() {
          const [markerTitle, setMarkerTitle] = React.useState(INC_POINT.title);
          const [markerTitle2, setMarkerTitle2] = React.useState('');

          React.useEffect(() => {
            let inc = 0;
            const updateTitle = () => {
              inc++;
              setMarkerTitle('Marker inc #' + inc);
            };

            updateTitle();
            const timer = setInterval(updateTitle, 1000);

            setTimeout(() => {
              setMarkerTitle2('Marker 2');
            }, 1000);

            return () => {
              clearInterval(timer);
            };
          }, []);

          return (
            <React.Fragment>
              <MMap location={LOCATION} ref={(x) => (map = x)}>
                <MMapDefaultSchemeLayer />
                <MMapDefaultFeaturesLayer />
                {POINTS.map((point, index) => {
                  if (point.element) {
                    return <MMapMarker key={index} {...point} markerElement={point.element(point)} />;
                  }

                  return <MMapDefaultMarker key={index} {...point} />;
                })}
                <MMapDefaultMarker coordinates={INC_POINT.coordinates} title={markerTitle} />
                <MMapDefaultMarker {...INC2_POINT} title={markerTitle2} subtitle={markerTitle2} />
                <MMapControls position="right">
                  <MMapZoomControl />
                </MMapControls>
              </MMap>
            </React.Fragment>
          );
        }

        ReactDOM.render(
          <React.StrictMode>
            <App />
          </React.StrictMode>,
          document.getElementById('app')
        );
      }
    </script>

    <!-- prettier-ignore -->
    <style> html, body, #app { width: 100%; height: 100%; margin: 0; padding: 0; font-family: Arial, Helvetica, sans-serif; } .toolbar { position: absolute; z-index: 1000; top: 0; left: 0; display: flex; align-items: center; padding: 16px; } .toolbar a { padding: 16px; }  </style>
    <link rel="stylesheet" href="./common.css" />
  </head>
  <body>
    <div id="app"></div>
  </body>
</html>
<!DOCTYPE html>
<html>
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1, minimum-scale=1" />
    <script crossorigin src="https://unpkg.com/vue@3/dist/vue.global.js"></script>
    <script src="https://js.api.mappable.world/v3/?apikey=<YOUR_APIKEY>&lang=en_US" type="text/javascript"></script>
    <script src="./common.js"></script>

    <script>
      window.map = null;

      main();
      async function main() {
        const [mappableVue] = await Promise.all([mappable.import('@mappable-world/mappable-vuefy'), mappable.ready]);
        const vuefy = mappableVue.vuefy.bindTo(Vue);
        const {MMap, MMapDefaultSchemeLayer, MMapDefaultFeaturesLayer, MMapMarker, MMapControls} =
          vuefy.module(mappable);
        const {MMapZoomControl} = vuefy.module(await mappable.import('@mappable-world/mappable-controls@0.0.1'));
        const {MMapDefaultMarker} = vuefy.module(await mappable.import('@mappable-world/mappable-markers@0.0.1'));

        const app = Vue.createApp({
          components: {
            MMap,
            MMapDefaultSchemeLayer,
            MMapDefaultFeaturesLayer,
            MMapMarker,
            MMapControls,
            MMapZoomControl,
            MMapDefaultMarker
          },
          setup() {
            const markerTitle = Vue.ref(INC_POINT.title);
            const markerTitle2 = Vue.ref('');
            const refMap = (ref) => {
              window.map = ref?.entity;
            };
            let timer;
            Vue.onMounted(() => {
              let inc = 0;
              const updateTitle = () => {
                inc++;
                markerTitle.value = 'Marker inc #' + inc;
              };
              updateTitle();
              const timer = setInterval(updateTitle, 1000);

              setTimeout(() => {
                markerTitle2.value = 'Marker 2';
              }, 1000);
            });
            Vue.onUnmounted(() => {
              clearInterval(timer);
            });
            return {LOCATION, POINTS, INC_POINT, INC2_POINT, refMap, markerTitle, markerTitle2};
          },
          template: `
                        <MMap :location="LOCATION" :ref="refMap">
                            <MMapDefaultSchemeLayer />
                            <MMapDefaultFeaturesLayer />
                            <template v-for="(point, index) in POINTS" :key="index">
                                <MMapMarker
                                    v-if="point.element"
                                    :coordinates="point.coordinates"
                                    :markerElement="point.element(point)" />
                                <MMapDefaultMarker
                                    v-else
                                    v-bind="point" />
                            </template>
                            <MMapDefaultMarker :coordinates="INC_POINT.coordinates" :title="markerTitle" />
                            <MMapDefaultMarker v-bind="INC2_POINT" :title="markerTitle2" :subtitle="markerTitle2" />
                            <MMapControls position="right">
                                <MMapZoomControl></MMapZoomControl>
                            </MMapControls>
                        </MMap>`
        });
        app.mount('#app');
      }
    </script>

    <!-- prettier-ignore -->
    <style> html, body, #app { width: 100%; height: 100%; margin: 0; padding: 0; font-family: Arial, Helvetica, sans-serif; } .toolbar { position: absolute; z-index: 1000; top: 0; left: 0; display: flex; align-items: center; padding: 16px; } .toolbar a { padding: 16px; }  </style>
    <link rel="stylesheet" href="./common.css" />
  </head>
  <body>
    <div id="app"></div>
  </body>
</html>