Skip to content

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

Совет

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

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

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

html
<yandex-map
    :height="height"
    :settings="{
        location: {
            center,
            zoom,
        },
        theme,
        showScaleInCopyrights: true,
    }"
    :width="width"
>
    <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"
            :position="`${ positionX } ${ positionY }` as any"
            :settings="point"
            :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,
    YandexMapControlButton,
    YandexMapControls,
    YandexMapDefaultFeaturesLayer,
    YandexMapDefaultMarker,
    YandexMapDefaultSchemeLayer,
    YandexMapMarker,
    YandexMapZoomControl,
} 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.