写在开头
哈喽!各位早上好呀,现2024年4月18日早上9点整,天气晴朗。☀
开始步入夏日了?最近几天都好闷热,每次来到公司基本都是”汗流浃背”,Em…可能小编有点虚?。。。不承认,好在公司已经开放空调了,凉快。❄
绿阴不减来时路,添得黄鹂四五声。
上周,写了篇 拖动❓元素拖动、列表拖动、表格拖动(列与行)🍊🍊🍊 的文章,反响好像还行,好久没有超过50个赞了🥳,感谢各位友友的宝贵一赞。👍
其实,很早之前,小编就开始进入佛系状态了,写文章随心所欲,想写就写,想写什么就写什么,不想写也就作罢,就…….很躺平👻,快乐无限。
不过呢,每个月还是会积极参与 金石计划征文活动 ,毕竟还能赚点小钱,补贴上每个月的交通费,何乐而不为呢,是吧是吧。而且这过程也能积累、记录、沉淀一些技术知识点,就挺好,这里也感谢掘金❤有这么一个活动来激励创作者,当然,也鼓励你、你们、他、他们快来参加叭。😉
扯远了!回到正题,本章将分享一些关于 Javascript
实用小功能,请诸君按需食用。(✪ω✪)
手动将文本复制到剪贴板
效果如下:
(开胃小菜,会的可以跳过,继续往下瞧。😐)
具体逻辑如下:
1 | html复制代码<!DOCTYPE html> |
原理嘛,也简单,利用了一个文本域,借用它的选中能力,再去执行复制API document.execcommand 就行。
不过这个API已经准备被弃用,虽然还能用,但是未来某天可能就GG掉了。🤡
已弃用: 不再推荐使用该特性。虽然一些浏览器仍然支持它,但也许已从相关的 web 标准中移除,也许正准备移除或出于兼容性而保留。请尽量不要使用该特性,并更新现有的代码;参见本页面底部的兼容性表格以指导你作出决定。请注意,该特性随时可能无法正常工作。
那么,替换方案呢?肯定有!
可以瞧瞧 阮一峰 大神的总结,很全面,肯定有你想要且喜欢的。
现有操作剪贴板一共有三种方式:
- document.execCommand
- clipboard API
- copy事件、paste事件、cut事件
而咱们的案例可以这样子修改:
1 | javascript复制代码button.addEventListener('click', (e) => { |
一行即可解决。👻
在复制的文本添加自定义内容
Em……这个是剪贴板的另一种常见应用形式,效果如下:
很简单啦,顺嘴一提。😗
1 | javascript复制代码<!DOCTYPE html> |
input随着输入内容自动增长-宽度
效果如下:
也是一种常见功能吧,我们来瞧瞧具体实现:
1 | html复制代码<!DOCTYPE html> |
整个过程原理大致是,将 input
实时输入的内容同步给一个容器,计算容器的宽度,再将容器宽度同步回 input
就能完成如上的效果啦。😁
这个小功能关键点是检测内容的宽度,而在 JS
中检测内容宽度一般用两种方式,其一,使用假元素,也就是我们上面使用的形式;另一种就是使用 canvas
的 measureText
方法。
大致过程如下:
1 | javascript复制代码function measureWidth(text, font) { |
textarea随着输入内容自动增长-高度
效果如下:
也是比较常见的功能了,具体实现如下:
1 | html复制代码<!DOCTYPE html> |
呃……两行代码就能完成😲,完全没难度,一个小技巧。
先将文本域的高度重置成 auto
,这样我们就能使用 scrollHeight
获取内容的实际高度,然后,将文本域的高度设置成 scrollHeight
的值,这就能让文本域自动扩展适配内容了。
将页面部分内容全屏化
效果如下:
具体实现:
1 | html复制代码<!DOCTYPE html> |
这个案例主要考验咱们对 JS
中的一些API使用是否熟练,可以自个瞅瞅哈。👻
- document.fullscreenEnabled:检测全屏模式是否可用。
- document.fullscreenElement:返回已经被全屏化的元素,如果没有则返回
null
。 - document.exitFullscreen:退出全屏。
- element.requestFullscreen:将元素全屏化,注意是异步的。
- fullscreenchange:监听浏览器进入或退出全屏模式后立即触发。
同步两个元素之间的滚动
这个案例源于小编提交 Git
代码时,查看相关代码的前后情况,那时编辑器的同步滚动引起了我的注意,就想着写个DEMO玩玩看。😗
同步滚动的应用情况还是非常常见的,如小编此时正在用的掘金Markdown编辑器、处理并排翻译项目时等等吧,总之,同步滚动可以给我们提供更便捷的工作效率。
下面咱们来一步一步实现这个小功能案例,先整上布局:
1 | html复制代码<!DOCTYPE html> |
Em…😐没啥难,就是通过 JS
动态随机生成了一些子元素,方便我们后续的测试,大概整出来的效果如下:
现在左右两边各自滚自己的,互不干扰。
而我们要如何来同步两边的滚动呢?很简单!咱们只要给两边加上监听器(scroll
),当用户滚动其中一边,咱们更新另一边就可以。
具体实现过程:
1 | javascript复制代码left.addEventListener('scroll', e => { |
效果:
我们用了 Element.scrollTo API 来进行滚动操作,看起来是不是还不错?😉
不过,这还没完,这里还存在两个问题🔉:
- 可能造成无限滚动循环:因为我们左右两边都监听了
scroll
事件,如果用户滚动了左边,我们去通过Element.scrollTo
API滚动右边,这里其实右边也会触发scroll
事件,那么它又会去同步滚动左边,这就会造成一个死循环的无限滚动了。而为什么动态图中看起来很正常呢?这是因为咱们在behavior
参数上使用了instant
值,如果你换成smooth
值,这个问题就比较容易复现出来。
- 总高度滚动不同步:仔细瞧动图,你会发现左边还没滚动到底的时候,右边就已经到底了,这可不符合我们同步滚动的需求呀❗造成这个原因是两边的总高度不一致,虽然现在子元素高度都是一样的,但是子元素个数是随机的,未来也可能是子元素高度不一样高,反正就是两边的总高度可能会不一样高,那么滚动就不可能完全同步。
对于这两个问题,咱们来逐一击破💣。
要解决无限滚动循环问题,我们可以先暂时将未滚动元素的事件监听器给先移除了,等滚动结束后再加回来,Em…说着…很简单😑,但…具体要如何做呢?
且看:
1 | javascript复制代码const boxs = [left, right]; |
这里用到了一个 requestAnimationFrame API来判断再次添加回事件监听器的时机,按它的API介绍”要求浏览器在下次重绘之前调用”,也就是它会在下一次重绘之前被调用。
而滚动操作咱们使用了 Element.scrollTo
API,它通常情况下,仅会引起重绘,因为滚动并不改变元素的布局。但是,如果滚动导致某些依赖于滚动位置的计算(如计算动态加载的内容或响应滚动事件而改变样式的元素)发生,那么它也可能间接引起回流,这时可能就要考虑加一个”宏任务”(setTimeout
)来判断时机了。
那么,无限滚动循环问题咱们就如此解决掉了😎,其实关键点是滚动结束时机的把控,这会涉及重绘、回流、事件循环等 JS
的知识点。
而另一个总高度滚动不同步问题呢❓改动不大,先直接贴上代码:
1 | javascript复制代码function syncScroll(scrolledElement, element) { |
开始咱们是直接将滚动距离 scrolledElement.scrollTop
同步到另一个未滚动元素上。
但是当元素具有不同的高度时,滚动位置可能会变得不同步。要解决这个问题,咱们只能计算出滚动比例再去推导出不同元素的实际需要滚动的距离。
好了,通过这些修改,我们的同步滚动功能现在可以处理具有不同数量块的多个可滚动元素,同时保持其滚动位置完美同步,完美收工。🥳
当前时间2024年4月18日下午5点22分,文章还没写完,快下班了😔,咋搞❗❗❗ 看来今天怕是写不完了,本来打算是随便水水得了😆,不知不觉中间又扩展很多出来,自己又写得慢吞吞,唉。
打印图片
图片打印听起来好像挺高大上😲,其实不然。
在浏览器中,提供了打印的方法 window.print(),这个方法不需要任何参数,直接调用即可。
不过,它是将整个网页进行打印,这倒是与我们只想打印图片的需求有点差异,不过问题不大,咱们可以利用一个 iframe
来解决,iframe
里面就放一张图片就行嘛,这也是当前浏览器上打印局部内容的主流方案。
就不卖关子了😄,直接上代码:
1 | html复制代码<!DOCTYPE html> |
就一个方法,标有详细的注解,这里就不多说啦,看就完事。👻
贴两张图瞅瞅叭。
整个网页打印:
只打印图片:
可调整大小的视图
效果如下:
看这个案例的具体实现之前,小编建议你先稍微瞅瞅另一篇文章 拖动❓元素拖动、列表拖动、表格拖动(列与行)🍊🍊🍊 ,看完之后,这个案例对你来说就是手到擒来的事情,没吹🙅,真是这样。
结构与样式:
1 | html复制代码<!DOCTYPE html> |
HTML+CSS
就没什么好说的了,主要咱们是来看看逻辑方面是如何做的。😀
1 | javascript复制代码document.addEventListener('DOMContentLoaded', () => { |
父窗口与iframe之间的通信
效果如下:
Em……我想作为一个前端切图仔,你多少也有所了解,当前,在 JS
中要实现父窗口与 iframe
之间的通信,我们基本都会选择 message
事件与 postMessage
方法相结合的形式。
只不过现在越来越少会使用 iframe
了吧😷, 毕竟还有更优的方案可以选择-微前端。
但是呢……Em……没事的话,还是可以学学着玩的。😉
咱们新建 index.html
与 iframe.html
两个页面,具体过程如下:
1 | html复制代码<!DOCTYPE html> |
1 | html复制代码<!DOCTYPE html> |
很简单,但也有一些注意事项:
- 总是在发送和接收消息时验证消息来源,以确保安全。
- 如果父窗口和
iframe
在同一域下,可以省略event.origin
的检查。 postMessage
方法的第二个参数(目标源)可以是具体的源(如一个域名),也可以是'*'
,表示任何源都可以,但这会带来安全隐患。
至此,本篇文章就写完啦,撒花撒花。
希望本文对你有所帮助,如有任何疑问,期待你的留言哦。
老样子,点赞+评论=你会了,收藏=你精通了。
本文转载自: 掘金