gpt4 book ai didi

javascript - 自定义上下文菜单始终可见

转载 作者:行者123 更新时间:2023-11-28 02:25:13 27 4
gpt4 key购买 nike

我正在实现一个自定义菜单,该菜单在用户单击鼠标左键时出现,但我在尝试设置菜单的位置 (X, Y) 时遇到了问题,以便无论在哪个位置都可以看到整个菜单它打开的页面的一部分。

下图代表问题:

a busy cat

菜单的宽度会随着文本的增加而增加,因此调整其位置和高度也是一个挑战。

var elements = $('#content').find('h1, p, span');

var setMenuPosition = function(x, y) {
$("#menu").css('top', y);
$("#menu").css('left', x);
};

var setSelectedText = function() {
$('#menu').data('text', $(this).text());
};

var openMenu = function(e) {
e.stopPropagation();
elements.css('border', '1px solid transparent');
$(this).css('border', '1px dashed #333');
$('#menu').addClass('active');
$('#selected-text').text($('#menu').data('text'));
setMenuPosition(e.pageX, e.pageY);
};

var closeMenu = function() {
elements.css('border', '1px solid transparent');
$('#menu').removeClass('active');
};

$('#content').find('h1, p, span').on('mouseenter', setSelectedText);
$('#content').find('h1, p, span').on("click", openMenu);
$('#menu').on('mouseleave', closeMenu);
h1,
p,
span {
border: 1px solid transparent;
}

#content {
background-color: #e9e9ea;
padding: 25px;
}

#menu {
visibility: hidden;
opacity: 0;
transition: visibility 0s, opacity 0.5s linear;
background-color: #84ce6a;
color: #fff;
padding: 15px;
position: absolute;
min-width: 200px;
border-radius: 8px;
}

#menu.active {
visibility: visible;
opacity: 1;
}

#my-span {
background-color: rgb(255, 79, 79);
color: rgb(255, 255, 255);
padding: 0px 5px;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>
<div id="content">
<h1>My Title</h1>
<p>My text</p>
<p>My another text</p>
<p>My text <span id="my-span">My span</span>, other part of the same text</p>
</div>

<div id="menu">
<h4>Selected text is: <span id="selected-text"></span></h4>
<button>
Ok
</button>
</div>

我的 fiddle :https://jsfiddle.net/robsonnogueira/295d78ak/

最佳答案

要防止菜单超出视口(viewport),您需要以下逻辑:

// 1. Set menu content
$menuContent.text(ev.currentTarget.textContent);

// 2. Get X, Y click coordinates
let X = ev.clientX;
let Y = ev.clientY;

// 3. Fix X, Y
X = Math.max(0, Math.min(X, $win.width() - $menu.outerWidth(true)) );
Y = Math.max(0, Math.min(Y, $win.height() - $menu.outerHeight(true)) );

// 4. Show menu
$menu.css({left:X, top:Y}).addClass('is-visible');

它考虑了窗口(viewport)大小和菜单大小(在其内容被插入之后)- 并通过使用 的组合相应地修复、修改 X、Y 坐标Math.max()Math.min()

这是一个例子:

jQuery($ => {

const $win = $(window);
const $menu = $('#menu');
const $menuContent = $('#menu-content');

const menuOpen = (ev) => {
ev.stopPropagation();

// 1. Set menu content
$menuContent.text(ev.currentTarget.textContent);

// 2. Get X, Y click coordinates
let X = ev.clientX;
let Y = ev.clientY;

// 3. Fix X, Y
X = Math.max(0, Math.min(X, $win.width() - $menu.outerWidth(true)) );
Y = Math.max(0, Math.min(Y, $win.height() - $menu.outerHeight(true)) );

// 4. Show menu
$menu.css({left:X, top:Y}).addClass('is-visible');
}

const menuClose = () => {
$menu.removeClass('is-visible');
}

// Events
$(".menu-open").on('click', menuOpen);
$(".menu-close").on('click', menuClose);
$(document).on('click', menuClose);
$menu.on('click', ev => ev.stopPropagation());

});
html, body {
height: 100%;
margin:0;
font: 14px/1.4 sans-serif;
}

#menu {
position: fixed;
max-width: 300px;
left: 0;
top: 0;
background: #84ce6a;
padding: 10px 20px;
visibility: hidden;
opacity: 0;
transition: visibility 0.24s, opacity 0.24s;
}

#menu.is-visible {
visibility: visible;
opacity: 1;
}

/*Demo only*/
.menu-open{
position: absolute;
}
.menu-open:nth-child(1) {top: 0; left: 0;}
.menu-open:nth-child(2) {top: 0; right: 0;}
.menu-open:nth-child(3) {bottom: 0; left: 0;}
.menu-open:nth-child(4) {bottom: 0; right: 0;}
<span class="menu-open">Click to open menu</span>
<span class="menu-open">Click me</span>
<span class="menu-open">Click here to open menu</span>
<span class="menu-open">Click to open menu</span>

<div id="menu">
<h3>This is my menu</h3>
<div id="menu-content"></div>
<button class="menu-close">CLOSE MENU</button>
</div>

<script src="//code.jquery.com/jquery-3.3.1.min.js"></script>

以上可以通过以下方式进一步改进:

  • 使菜单在“右侧”上的表现与在左侧类似——首先计算点翻转是否可实现(将元素锚定在对面的右上角,右-底部或左下角),或者像现在一样坚持到远边。
  • 如果不适合视口(viewport),则修复菜单宽度/高度(可能需要为菜单内容/正文滚动条添加一些额外的 CSS)

关于javascript - 自定义上下文菜单始终可见,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54655646/

27 4 0
文章推荐: javascript - 强制从数据列表或等效项中选择(文本输入)
文章推荐: html - 有没有办法阻止 Outlook 破坏