我使用 css @keyframes 创建了圆形的基本动画。
我正在使用 javascript 通过单击圆圈内来触发动画开始/停止。
动画本身可以分为 5 个(循环)阶段:
暂停-扩展-暂停-收缩-暂停(参见下面的@keyframes css部分)
我想要实现的目标最终是能够设置动画持续时间并更改关键帧的值(例如通过具有暂停和扩展/收缩持续时间的输入字段 - 细节对于这个问题的范围并不重要) 。我已经组合了一个 JavaScript 函数来执行此任务,并将其设置为onload
只是为了测试它是如何工作的。
My HTML:
<!doctype html>
<html>
<head>
<meta content="text/html;charset=utf-8" http-equiv="Content-Type">
<meta content="utf-8" http-equiv="encoding">
<link rel="stylesheet" href="style.css">
<script src = "animation.js"></script>
</head>
<body onload=setAnimationDuration(1,1)>
<div id="circle" class='circle-paused' onclick=cssAnimation()></div>
</body>
</html>
My CSS:
#circle {
position: absolute;
top: 50%;
left: 50%;
transform: translateX(-50%) translateY(-50%);
}
.circle-paused {
width: 9%;
padding-top: 9%;
border-radius: 50%;
background-color: #800080;
margin: auto;
}
.circle-animated {
/* 2 sec pause 4 sec expand_shrink*/
width: 9%;
padding-top: 9%;
-webkit-animation-name: my-circle; /* Safari 4.0 - 8.0 */
-webkit-animation-duration: 12s; /* Safari 4.0 - 8.0 */
animation-name: my-circle;
animation-duration: 12s;
animation-iteration-count: infinite;
animation-timing-function: linear;
border-radius: 50%;
margin: auto;
}
@keyframes my-circle {
0% {background-color: #800080; width: 9%; padding-top: 9%;}
33.3% {background-color: #D8BFD8; width: 28%; padding-top: 28%;}
50% {background-color: #D8BFD8; width: 28%; padding-top: 28%;}
83.3% {background-color: #800080; width: 9%; padding-top: 9%;}
100% {background-color: #800080; width: 9%; padding-top: 9%;}
}
我的 JavaScript:
function cssAnimation() {
if (document.getElementById('circle').className == 'circle-paused') {
document.getElementById('circle').className = 'circle-animated'
} else {
document.getElementById('circle').className = 'circle-paused'
}
}
function findKeyframes(animation_name) {
// get list of current keyframe rules
var style_sheet = document.styleSheets;
for (var i = 0; i < style_sheet.length; ++i) {
for (var j = 0; j < style_sheet[i].cssRules.length; ++j) {
// type 7 correspond to CSSRule.KEYFRAMES_RULE, for more info see https://developer.mozilla.org/en-US/docs/Web/API/CSSRule
if (style_sheet[i].cssRules[j].type == 7 && style_sheet[i].cssRules[j].name == animation_name) {
return style_sheet[i].cssRules[j];
}
}
}
// keyframe rules were not found for given animation_name
return null;
}
function getPercentage(total, fraction) {
// Returns what percentage the fraction is from total
// The result is rounded to 1 decimal place
return Math.round(((100 / total) * fraction) * 10) / 10;
}
function setAnimationDuration(pause, expand_shrink) {
var total_animation_duration = (pause * 2) + (expand_shrink * 2)
var pause_percentage = getPercentage(total_animation_duration, pause)
var expand_shrink_percentage = getPercentage(total_animation_duration, expand_shrink)
var pause1 = pause_percentage + expand_shrink_percentage;
var shrink = pause1 + expand_shrink_percentage;
var frame_percentage_list = [0, expand_shrink_percentage, pause1, shrink, 100]
var key_frame_list = findKeyframes('my-circle')
var new_rule_list = []
var to_be_removed_key_list = []
//create array of new rules to be inserted
//collecting old keys of rules to be deleted
for(var i = 0; i < key_frame_list.cssRules.length; i++) {
var current_rule = key_frame_list.cssRules[i].cssText
to_be_removed_key_list.push(key_frame_list.cssRules[i].keyText)
new_rule_list.push(current_rule.replace(/[+-]?([0-9]*[.])?[0-9]+%/, frame_percentage_list[i] + '%'))
}
// delete old rules
for(var i = 0; i < to_be_removed_key_list.length; i++) {
key_frame_list.deleteRule(to_be_removed_key_list[i])
}
// populate new ruels
for(var i = 0; i < new_rule_list.length; i++) {
key_frame_list.appendRule(new_rule_list[i])
}
document.getElementById('circle').style.animationDuration = total_animation_duration + "s"
}
上面的代码,在 JSFiddle 上 https://jsfiddle.net/mfo4qk9dv2ua5iksttbnqx64c/rpve2e9L/
问题本身:
该代码在 FireFox (55.0.3)、Chrome (61.0) 和 Safari (11.0) 中按预期工作。不过,当我开始在 IE11 中测试它时,我发现key_frame_list.deleteRule('rule_key')
抛出一个Invalid argument
错误。在研究这个问题时,我发现(并陷入困境)本文 http://usefulangle.com/post/39/adding-css-to-stylesheet-with-javascript(虽然它没有解决 IE 问题,但它提高了我对 css 动画的整体理解)。在 MSDN 上我找到了两个参考文献,涉及deleteRule
: one https://msdn.microsoft.com/en-us/library/ff975162(v=vs.85).aspx and two https://msdn.microsoft.com/en-us/library/hh772386(v=vs.85).aspx。虽然我不太明白这是什么意思,但第二个 https://msdn.microsoft.com/en-us/library/hh772386(v=vs.85).aspx, by:
键必须解析为 0 到 1 之间的数字,否则规则将被忽略。
我假设在 IE 中你必须将索引传递给deleteRule
而不是字符串键。所以我尝试在 IE 控制台中检查我的假设。这是我发现的(鉴于我的 js 代码位于 onload 中):
var key_frame_list = findKeyframes('my-circle')
key_frame_list.cssRules.length => 5
key_frame_list.deleteRule(0)
key_frame_list.cssRules.length => 4
key_frame_list.deleteRule(1)
key_frame_list.cssRules.length => 3
key_frame_list.deleteRule(0)
key_frame_list.deleteRule(1)
key_frame_list.deleteRule(2)
...
key_frame_list.cssRules.length => 3
正在发生的事情是:
key_frame_list.deleteRule(0)
- 删除第一条规则(即 0%)
key_frame_list.deleteRule(1
) - 删除最后一条规则(即 100%)
之后无论我传递到哪个索引key_frame_list.deleteRule()
the key_frame_list.cssRules.length
仍为 3。
我的期望是我能够再次出现key_frame_list.deleteRule(0)
并删除所有规则(正如我预期的那样,索引会在每次规则删除后发生变化)。
现在,我想了解:
- 正确的使用方法是什么(基本上是“我做错了什么吗?”)
deleteRule
在 IE 中(或者是否应该使用其他方法)?
- 为什么我无法删除五个规则中的两个以上?
- 是否有一种适合此目的的方法可以在 Firefox、Chrome 和 IE11 上使用相同的参数,我不知道?