Skip to content

Добавление маркеров на карту

Совет

Чтобы поведение обычных маркеров соответствовало Yandex Maps 2, задайте:

  • positionX на left-center
  • positionY на top.

position будет сформирован как left-center top.

html
<yandex-map
  :settings="{
    location: {
      center,
      zoom,
    },
    theme,
    showScaleInCopyrights: true,
  }"
  :width="width"
  :height="height"
>
  <yandex-map-default-scheme-layer />
  <yandex-map-default-features-layer />
  <yandex-map-controls :settings="{ position: 'right' }">
    <yandex-map-zoom-control />
  </yandex-map-controls>

  <yandex-map-controls :settings="{ position: 'top left' }">
    <yandex-map-control-button>
      <label>
        Position (X)

        <select v-model="positionX">
          <option v-for="(key, value) in positionsX" :key="key" :value="value">
            {{ key }}
          </option>
        </select>
      </label>
    </yandex-map-control-button>
    <yandex-map-control-button>
      <label>
        Position (Y)

        <select v-model="positionY">
          <option v-for="(key, value) in positionsY" :key="key" :value="value">
            {{ key }}
          </option>
        </select>
      </label>
    </yandex-map-control-button>
  </yandex-map-controls>

  <template v-for="(point, index) in POINTS" :key="index">
    <yandex-map-marker
      v-if="'element' in point"
      :settings="point"
      :position="`${positionX} ${positionY}` as any"
      :style="{
        '--color': 'color' in point && point.color,
        '--image': 'colors' in point && diagramBackground(point.colors),
      }"
    >
      <template v-if="point.element === 'diagram'">
        <div
          v-if="'title' in point"
          class="pie-marker-title"
        >
          {{ point.title }}
        </div>
        <div
          class="pie-marker"
        />
      </template>
      <div
        v-else-if="point.element === 'circle'"
        class="circle"
        :style="{
          '--radius': 'radius' in point ? point.radius : '20px',
          '--color': 'color' in point ? point.color : undefined,
          '--icon': 'icon' in point ? point.icon : '#fff',
          '--image': 'icon' in point ? point.icon : undefined,
        }"
        :title="'title' in point && point.title"
      >
        <div class="circle_element" />
      </div>
      <div
        v-else-if="point.element === 'icon'"
        class="icon"
        :style="{
          '--size': 'size' in point ? point.size : '20px',
          '--color': 'color' in point ? point.color : undefined,
          '--icon': 'icon' in point ? `url(${point.icon})` : undefined,
        }"
      >
        <div
          v-if="'title' in point"
          class="icon_title"
          v-html="point.title"
        />
      </div>
    </yandex-map-marker>
    <yandex-map-default-marker
      v-else
      :settings="point"
    />
  </template>
  <yandex-map-default-marker :settings="{ coordinates: INC_POINT.coordinates, title: markerTitle }" />
</yandex-map>
ts
import {
  YandexMap,
  YandexMapControls,
  YandexMapDefaultFeaturesLayer,
  YandexMapDefaultMarker,
  YandexMapDefaultSchemeLayer,
  YandexMapMarker,
  YandexMapControlButton,
  YandexMapZoomControl,
} from 'vue-yandex-maps';
import type { YandexMapMarkerPosition } from 'vue-yandex-maps';
import { onMounted, onUnmounted, ref } from 'vue';
import type { LngLat } from '@yandex/ymaps3-types';

type PartialRecord<K extends keyof any, T> = {
  [P in K]?: T;
};

const INC_POINT = {
  coordinates: [37.95, 55.9] as LngLat,
  title: 'Marker inc #0',
};

const markerTitle = ref('');

const positionsX = {
  left: 'left',
  right: 'right',
  'right-center': 'right-center',
  'left-center': 'left-center',

  custom: '-25%',

  default: 'default',
} satisfies PartialRecord<YandexMapMarkerPosition | 'custom', string>;

const positionsY = {
  top: 'top',
  bottom: 'bottom',
  'top-center': 'top-center',
  'bottom-center': 'bottom-center',

  custom: '-25%',

  default: 'default',
} satisfies PartialRecord<YandexMapMarkerPosition | 'custom', string>;

const positionX = ref<keyof typeof positionsX>('default');
const positionY = ref<keyof typeof positionsY>('default');

onMounted(() => {
  let inc = 0;
  const updateTitle = () => {
    inc++;
    markerTitle.value = `Marker inc #${inc}`;
  };
  updateTitle();
  const timer = setInterval(updateTitle, 1000);
  onUnmounted(() => {
    clearInterval(timer);
  });
});

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

  return `conic-gradient(${gradient.join(', ')})`;
};

const POINTS: any[] = [
  { coordinates: [37.8, 55.8] },
  {
    coordinates: [37.6, 55.847],
    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: [37.738521, 55.684758],
    color: '#0095b6',
    title: 'color <strong>bondi beach water<strong>',
    draggable: true,
  },
  {
    coordinates: [37.715175, 55.833436],
    color: '#735184',
    title: '<strong>Silver crimson<strong> color',
    draggable: true,
  },
  {
    coordinates: [37.529789, 55.687086],
    color: '#3caa3c',
    title: 'love toad color',
    draggable: true,
    element: 'circle',
  },
  {
    coordinates: [37.95, 55.782392],
    color: 'yellow',
    title: 'color <strong>sun<strong>',
    draggable: true,
    onClick: () => alert('click'),
  },
  {
    coordinates: [37.656123, 55.642063],
    title: 'color <strong>red<strong>',
    size: '60px',
    icon: '',
    draggable: true,
    element: 'icon',
    onDoubleClick: () => alert('Double click'),
  },
  {
    coordinates: [37.487208, 55.826479],
    title: 'the color of <strong>Pacific Ocean<strong>',
    color: '#3b5998',
    draggable: true,
    mapFollowsOnDrag: true,
  },
  {
    coordinates: [37.435023, 55.694843],
    color: '#477510',
    title: 'nose color Donatello',
    subtitle: 'Very long but incredibly interesting text',
    draggable: true,
  },
  {
    coordinates: [37.535023, 55.6],
    color: '#343d44',
    title: 'Hello!',
    subtitle: 'Very long but <br>incredibly interesting text',
    draggable: true,
  },
  {
    coordinates: [37.814052, 55.790139],
    title: 'blue color',
    icon: 'url()',
    color: '#51aabd',
    radius: '50px',
    draggable: true,
    element: 'circle',
    onFastClick: () => alert('Fast click'),
  },
];
css
<style scoped>
select {
  border: 1px solid #000;
  padding-left: 5px;
}

.pie-marker-title {
  position: absolute;
  top: 120%;
  left: 50%;
  padding: 2px 4px;
  background-color: #fff;
  transform: translateX(-50%);
  color: var(--color);
}

.pie-marker {
  background-color: currentColor;
  background-image: var(--image);
  width: 50px;
  height: 50px;
  border-radius: 50%;
  overflow: hidden;
}

.circle {
  width: var(--radius);
  height: var(--radius);
  border-radius: 50%;
  background-color: currentColor;
  color: var(--color);
}

.circle_element {
  position: absolute;
  top: 50%;
  left: 50%;
  display: inline-block;
  width: 50%;
  height: 50%;
  border-radius: 50%;
  background: #fff var(--image) no-repeat center center;
  transform: translate3d(-50%, -50%, 0);
}

.icon {
  position: relative;
  width: var(--size);
  height: var(--size);
  color: var(--color);
  background: var(--icon) no-repeat center center / contain;
}

.icon_title {
  position: absolute;
  top: 120%;
  left: 50%;
  padding: 2px 4px;
  background-color: #fff;
  transform: translateX(-50%);
}
</style>

Сделано с ♥ под лицензией MIT.