diff --git a/src/components/Control/SwipeControl/demos/default.tsx b/src/components/Control/SwipeControl/demos/default.tsx
new file mode 100644
index 00000000..d26563cd
--- /dev/null
+++ b/src/components/Control/SwipeControl/demos/default.tsx
@@ -0,0 +1,35 @@
+import { LarkMap, RasterLayer, SwipeControl, useLayer } from '@antv/larkmap';
+import React from 'react';
+
+const CustomComponent = () => {
+ const rightRasterLayer = useLayer('rightRasterLayer');
+ return (
+
+ );
+};
+
+export default () => {
+ return (
+
+
+
+
+
+ );
+};
diff --git a/src/components/Control/SwipeControl/index.md b/src/components/Control/SwipeControl/index.md
new file mode 100644
index 00000000..e18a3f5b
--- /dev/null
+++ b/src/components/Control/SwipeControl/index.md
@@ -0,0 +1,37 @@
+---
+toc: content
+order: 2
+group:
+ title: 控件组件
+ order: 2
+nav:
+ title: 组件
+ path: /components
+---
+
+## 卷帘 - SwipeControl
+
+### 介绍
+
+该控件用于分屏对比两个地图上叠加图层,图层可以添加到地图的左侧(顶部)或右侧(底部)。未添加到卷帘上的图层将显示在两侧。
+
+### 代码演示
+
+
+
+### API
+
+| 参数 | 说明 | 类型 |
+| --- | --- | --- |
+| orientation | 卷帘方向设置,默认 'vertical' | `vertical|horizontal` |
+| ratio | 卷帘的位置,值域为 0 到 1, 默认正中间为 0.5 | `number` |
+| layers | 卷帘左侧的图层 | `Array` |
+| rightLayers | 卷帘左侧的图层 | `Array` |
+| className | 自定义样式名 | `string` |
+| style | 自定义样式 | `CSSProperties` |
+| onAdd | 组件被添加时的回调 | `(this) => void` |
+| onRemove | 组件被移除时的回调 | `(this) => void` |
+| onShow | 组件显示时的回调 | `(this) => void` |
+| onHide | 组件隐藏时的回调 | `(this) => void` |
+| onMoving | 卷帘移动事件 | `(data: {size: number[], ratio: number[]}) => void` |
+
diff --git a/src/components/Control/SwipeControl/index.tsx b/src/components/Control/SwipeControl/index.tsx
new file mode 100644
index 00000000..dc00f147
--- /dev/null
+++ b/src/components/Control/SwipeControl/index.tsx
@@ -0,0 +1,89 @@
+import type { IZoomControlOption } from '@antv/l7';
+import { Swipe as L7Swipe } from '@antv/l7';
+import { omitBy } from 'lodash-es';
+import React, { useEffect, useMemo, useState } from 'react';
+import { getStyleText } from '../../../utils';
+import { useLayerList, useScene } from '../../LarkMap/hooks';
+import { useL7ComponentEvent, useL7ComponentUpdate } from '../hooks';
+import type { SwipeControlProps } from './types';
+
+export const SwipeControl: React.FC = ({
+ onShow,
+ onHide,
+ onAdd,
+ onRemove,
+ onMoving,
+ className,
+ style,
+ rightLayers,
+ layers: layerItems,
+ ratio,
+ orientation,
+}) => {
+ const scene = useScene();
+ const [control, setControl] = useState();
+ const styleText = useMemo(() => getStyleText(style), [style]);
+ const fullLayerList = useLayerList();
+
+ const layers = (defaultLayers) => {
+ return defaultLayers?.length
+ ? defaultLayers
+ .map((layerItem) => {
+ if (typeof layerItem === 'string') {
+ return fullLayerList.find((layer) => layer.id === layerItem);
+ }
+ if (!Object.prototype.hasOwnProperty.call(layerItem, 'isComposite')) {
+ //@ts-ignore
+ const { layer } = layerItem;
+
+ if (typeof layer === 'string') {
+ const targetLayer = fullLayerList.find((item) => item.id === layer);
+ return targetLayer ? { ...layerItem, layer: targetLayer } : undefined;
+ }
+ return layerItem;
+ }
+
+ return layerItem;
+ })
+ .filter((layer) => !!layer)
+ .map((layerItem) => {
+ //@ts-ignore
+ return layerItem.subLayers ? layerItem.subLayers.getLayers().map((layer) => layer.layer) : layerItem.layer;
+ })
+ .flat()
+ : fullLayerList;
+ };
+
+ const controlOptions: Partial = useMemo(() => {
+ return {
+ rightLayers: layers(rightLayers),
+ layers: layers(layerItems),
+ ratio,
+ orientation,
+ style: styleText,
+ };
+ }, [className, styleText, rightLayers, layers, ratio, orientation]);
+
+ useEffect(() => {
+ const swipe = new L7Swipe(omitBy(controlOptions, (value) => value === undefined));
+ setControl(swipe);
+ scene.addControl(swipe);
+ return () => {
+ scene.removeControl(swipe);
+ setControl(undefined);
+ };
+ // eslint-disable-next-line react-hooks/exhaustive-deps
+ }, []);
+
+ useL7ComponentUpdate(control, controlOptions);
+
+ useL7ComponentEvent(control, {
+ add: onAdd,
+ remove: onRemove,
+ show: onShow,
+ hide: onHide,
+ moving: onMoving,
+ });
+
+ return <>>;
+};
diff --git a/src/components/Control/SwipeControl/types.ts b/src/components/Control/SwipeControl/types.ts
new file mode 100644
index 00000000..0398c0bc
--- /dev/null
+++ b/src/components/Control/SwipeControl/types.ts
@@ -0,0 +1,11 @@
+import type { ISwipeControlOption, Zoom as L7Zoom } from '@antv/l7';
+import type { CSSProperties } from 'react';
+import type { IControlCallback, Layer } from '../../../types';
+
+export interface SwipeControlProps
+ extends Omit, 'style' | 'layers'>,
+ IControlCallback {
+ style?: CSSProperties;
+ onMoving?: (data: { size: number[]; ratio: number[] }) => void;
+ layers: (Layer | string)[];
+}
diff --git a/src/components/Control/index.ts b/src/components/Control/index.ts
index aca7533a..1c889645 100644
--- a/src/components/Control/index.ts
+++ b/src/components/Control/index.ts
@@ -29,3 +29,6 @@ export type { ZoomControlProps } from './ZoomControl/types';
export { LogoControl } from './LogoControl';
export type { LogoControlProps } from './LogoControl/types';
+
+export { SwipeControl } from './SwipeControl';
+export type { SwipeControlProps } from './SwipeControl/types';
\ No newline at end of file