搜索

首页  >  问答  >  正文

使用 React 在图像上放置交互式标记

请帮忙。 我有一个像下图这样的图像显示在主页上(一个更大的图像)。 请注意,“FORGE”房屋的中间有一个白色和蓝色标记,并带有黑色箭头

“出租”房屋也是如此。

这些标记中的每一个都将充当交互式按钮。以下是预期的交互行为:

还希望它能够响应,如果当屏幕调整大小时,按钮应保留在这些特定位置。

我正在尝试使用绝对定位、x、y 坐标来尝试使其工作。但由于我喜欢此类用户界面的经验,无法找到可行的解决方案。不确定我是否应该使用画布或其他方法。

任何帮助将不胜感激。

我的代码看起来像这样,但看起来我没有走在正确的轨道上:

const ImageComponent = () => {

  const markers = [
    { name: 'Forge', x: 100, y: 200 },
    { name: 'Rentals', x: 300, y: 150 },
    // Add more 
  ];

  const handleMarkerClick = (m) => {
    // do something with marker
  };

  return (
    <div style={{ position: 'relative' }}>
      <img src="path/to/image.jpg" alt="Image with markers" />

      {markers.map((marker, index) => (
        <div
          key={index}
          className="marker"
          style={{ left: marker.x, top: marker.y, position: "absolute" }}
          onClick={() => handleMarkerClick(marker)}
        />
      ))}
    </div>
  );
};

P粉896751037P粉896751037447 天前822

全部回复(1)我来回复

  • P粉412533525

    P粉4125335252023-09-09 17:37:18

    您应该将控件放置在图像容器的绝对位置。 实现取决于多种因素,例如图像放置的位置、是否全屏显示、图像之前或之后是否有内容等。 但这段代码应该向您展示主要原理。

    如果在调整窗口大小时调整图像大小,则应该创建一个与图像完全相同的容器。然后,您可以设置控件相对于该容器的位置,并使它们固定在各自的点上。

    .container {
      width: 60vw; /* set 100vw to fit image window width*/
      position: relative;
    }
    .container img { width: 100%; display: block;}
    .container .control {
      position: absolute;
      width: 7.5%;  /* size in percents to resize controls with image */
      height: 7%;
      border-radius: 200px;
      background: red;
      transition: all 0.2s ease-in-out;
    }
    
    .container .control:hover {
      width: 20%;
    }
    
    #c1 { left: 50.4%; top: 26%; } /* position in percents, not in pixels */
    #c2 { left: 58.5%; top: 76.3%; }
    <div class="container">
      <img src="https://i.stack.imgur.com/BVaY9.jpg"/>
      <div class="control" id="c1"></div>
      <div class="control" id="c2"></div>
    </div>

    使用你的代码,它会是这样的(但我建议在样式文件中移动你的样式,学习如何在react中使用模块样式):

    const ImageComponent = () => {
    
      const markers = [
        { name: 'Forge', x: 50.4, y: 26 },
        { name: 'Rentals', x: 58.5, y: 76.3 },
        // Add more 
      ];
    
      const handleMarkerClick = (m) => {
        // do something with marker
      };
    
      return (
        <div style={{ position: 'relative', width: '100vw' }}>
          <img style={{ width: '100%', display: 'block'}} src="path/to/image.jpg" alt="Image with markers" />
    
          {markers.map((marker, index) => (
            <div
              key={index}
              className="marker"
              style={{ left: `${marker.x}%`, top: `${marker.y}%`, position: "absolute" }}
              onClick={() => handleMarkerClick(marker)}
            />
          ))}
        </div>
      );
    };
    

    只需修复位置并添加正常的控制视图即可。

    为了进行控制,您应该制作单独的组件。以下是如何实现您想要的行为的示例:

    .control {
      background: #08f;
      padding: 10px;
      width: 60px;
      box-sizing: border-box;
      border-radius: 200px;
      transition: all 0.2s ease-in-out;
      display: flex;
      align-items: center;
      overflow: hidden;
      cursor: pointer;
    }
    
    .control:hover {
      width: 200px;
    }
    
    .text {
      color: white;
      font: 30px Arial;
      margin-left: 16px;
      opacity: 0;
      transition: opacity 0.2s ease-in-out;
    }
    
    .control:hover .text {
      opacity: 1;
    }
    
    .icon {
      width: 40px;
      min-width: 40px;
      height: 40px;
      border-radius: 50%;
      box-shadow: 2px 2px 7px #0005;
      background-color: white;
      background-image: url('');
      background-size: 60%;
      background-repeat: no-repeat;
      background-position: center;
    }
    <div class="control">
      <div class="icon"></div>
      <div class="text">Home</div>
    </div>

    回复
    0
  • 取消回复