2020-09-09 00:40:22 +08:00
|
|
|
class TimelineMarker {
|
|
|
|
constructor(data) {
|
|
|
|
this.time = 0;
|
|
|
|
this.color = 0;
|
|
|
|
if (data) {
|
|
|
|
this.extend(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
extend(data) {
|
|
|
|
Merge.number(this, data, 'color');
|
|
|
|
Merge.number(this, data, 'time');
|
|
|
|
}
|
|
|
|
callPlayhead() {
|
|
|
|
Timeline.setTime(this.time)
|
|
|
|
Animator.preview()
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
showContextMenu(event) {
|
|
|
|
this.menu.open(event, this);
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
TimelineMarker.prototype.menu = new Menu([
|
|
|
|
{icon: 'flag', color: markerColors[0].standard, name: 'cube.color.'+markerColors[0].name, click: function(marker) {marker.color = 0;}},
|
|
|
|
{icon: 'flag', color: markerColors[1].standard, name: 'cube.color.'+markerColors[1].name, click: function(marker) {marker.color = 1;}},
|
|
|
|
{icon: 'flag', color: markerColors[2].standard, name: 'cube.color.'+markerColors[2].name, click: function(marker) {marker.color = 2;}},
|
|
|
|
{icon: 'flag', color: markerColors[3].standard, name: 'cube.color.'+markerColors[3].name, click: function(marker) {marker.color = 3;}},
|
|
|
|
{icon: 'flag', color: markerColors[4].standard, name: 'cube.color.'+markerColors[4].name, click: function(marker) {marker.color = 4;}},
|
|
|
|
{icon: 'flag', color: markerColors[5].standard, name: 'cube.color.'+markerColors[5].name, click: function(marker) {marker.color = 5;}},
|
|
|
|
{icon: 'flag', color: markerColors[6].standard, name: 'cube.color.'+markerColors[6].name, click: function(marker) {marker.color = 6;}},
|
|
|
|
{icon: 'flag', color: markerColors[7].standard, name: 'cube.color.'+markerColors[7].name, click: function(marker) {marker.color = 7;}},
|
|
|
|
{icon: 'delete', name: 'generic.delete', click: function(marker) {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (Animation.selected) Animation.selected.markers.remove(marker);
|
2020-09-09 00:40:22 +08:00
|
|
|
}}
|
|
|
|
])
|
|
|
|
|
|
|
|
const Timeline = {
|
|
|
|
animators: [],
|
2020-09-26 04:06:03 +08:00
|
|
|
selected: Keyframe.selected,//frames
|
2020-09-09 00:40:22 +08:00
|
|
|
playing_sounds: [],
|
|
|
|
playback_speed: 100,
|
|
|
|
time: 0,
|
|
|
|
get second() {return Timeline.time},
|
2020-09-15 20:21:04 +08:00
|
|
|
get animation_length() {return Animation.selected ? Animation.selected.length : 0;},
|
2020-09-09 00:40:22 +08:00
|
|
|
playing: false,
|
|
|
|
selector: {
|
|
|
|
start: [0, 0],
|
|
|
|
selecting: false,
|
|
|
|
selected_before: [],
|
|
|
|
down(e) {
|
|
|
|
if (e.which !== 1 || (
|
|
|
|
!e.target.classList.contains('keyframe_section') &&
|
|
|
|
!e.target.classList.contains('animator_head_bar') &&
|
|
|
|
e.target.id !== 'timeline_body_inner'
|
|
|
|
)) {
|
|
|
|
return
|
|
|
|
};
|
|
|
|
|
|
|
|
Timeline.selector.interval = setInterval(Timeline.selector.move, 1000/60);
|
|
|
|
document.addEventListener('mouseup', Timeline.selector.end, false);
|
|
|
|
|
|
|
|
var offset = $('#timeline_body_inner').offset();
|
|
|
|
var R = Timeline.selector;
|
|
|
|
R.panel_offset = [
|
|
|
|
offset.left,
|
|
|
|
offset.top,
|
|
|
|
]
|
|
|
|
R.start = [
|
|
|
|
e.clientX - R.panel_offset[0],
|
|
|
|
e.clientY - R.panel_offset[1],
|
|
|
|
]
|
|
|
|
if (e.shiftKey) {
|
|
|
|
Timeline.selector.selected_before = Timeline.selected.slice();
|
|
|
|
}
|
|
|
|
R.selecting = true;
|
|
|
|
$('#timeline_selector').show()
|
|
|
|
Timeline.selector.move(e)
|
|
|
|
},
|
|
|
|
move(e) {
|
|
|
|
var R = Timeline.selector;
|
|
|
|
if (!R.selecting) return;
|
|
|
|
//CSS
|
|
|
|
var offset = $('#timeline_body_inner').offset();
|
|
|
|
R.panel_offset = [
|
|
|
|
offset.left,
|
|
|
|
offset.top,
|
|
|
|
]
|
|
|
|
var rect = getRectangle(R.start[0], R.start[1], mouse_pos.x - R.panel_offset[0], mouse_pos.y - R.panel_offset[1])
|
|
|
|
$('#timeline_selector')
|
|
|
|
.css('width', rect.x + 'px')
|
|
|
|
.css('height', rect.y + 'px')
|
|
|
|
.css('left', rect.ax + 'px')
|
|
|
|
.css('top', rect.ay + 'px');
|
|
|
|
//Keyframes
|
|
|
|
var epsilon = 6;
|
2020-09-28 18:55:45 +08:00
|
|
|
let {channels} = Timeline.vue._data;
|
2020-09-09 00:40:22 +08:00
|
|
|
rect.ax -= epsilon;
|
|
|
|
rect.ay -= epsilon;
|
|
|
|
rect.bx += epsilon;
|
|
|
|
rect.by += epsilon;
|
|
|
|
|
|
|
|
var min_time = (rect.ax-Timeline.vue._data.head_width-8)/Timeline.vue._data.size;
|
|
|
|
var max_time = (rect.bx-Timeline.vue._data.head_width-8)/Timeline.vue._data.size;
|
|
|
|
|
|
|
|
Timeline.selected.empty()
|
|
|
|
for (var animator of Timeline.animators) {
|
|
|
|
var node = $('#timeline_body_inner .animator[uuid=' + animator.uuid + ']').get(0)
|
|
|
|
var offset = node && node.offsetTop;
|
|
|
|
for (var kf of animator.keyframes) {
|
|
|
|
if (Timeline.selector.selected_before.includes(kf)) {
|
|
|
|
Timeline.selected.push(kf);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
kf.selected = false;
|
|
|
|
if (kf.time > min_time &&
|
|
|
|
kf.time < max_time &&
|
2020-09-28 18:55:45 +08:00
|
|
|
channels[kf.channel] != false &&
|
|
|
|
(!channels.hide_empty || animator[kf.channel].length)
|
2020-09-09 00:40:22 +08:00
|
|
|
) {
|
2020-09-28 18:55:45 +08:00
|
|
|
var channel_index = 0 //animator.channels.indexOf(kf.channel);
|
|
|
|
|
|
|
|
for (var channel of animator.channels) {
|
|
|
|
if (kf.channel == channel) break;
|
|
|
|
if (channels[channel] != false && (!channels.hide_empty || animator[channel].length)) {
|
2020-09-09 00:40:22 +08:00
|
|
|
channel_index++;
|
|
|
|
}
|
|
|
|
}
|
2020-09-28 18:55:45 +08:00
|
|
|
|
2020-09-09 00:40:22 +08:00
|
|
|
height = offset + channel_index*24 + 36;
|
|
|
|
if (height > rect.ay && height < rect.by) {
|
|
|
|
kf.selected = true;
|
|
|
|
Timeline.selected.push(kf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//Scroll body
|
|
|
|
var body = $('#timeline_body').get(0)
|
|
|
|
var body_inner = $('#timeline_body_inner').get(0)
|
|
|
|
|
|
|
|
var top = mouse_pos.y - R.panel_offset[1] - body.scrollTop;
|
|
|
|
var bot = body.scrollTop + body.clientHeight - (mouse_pos.y - R.panel_offset[1]);
|
|
|
|
var lef = mouse_pos.x - R.panel_offset[0] - body.scrollLeft - Timeline.vue._data.head_width;
|
|
|
|
var rig = body.clientWidth - (mouse_pos.x - R.panel_offset[0] - body.scrollLeft);
|
|
|
|
|
|
|
|
let speed = 15;
|
|
|
|
|
|
|
|
if (top < 0) body.scrollTop = body.scrollTop - speed;
|
|
|
|
if (bot < 0) body.scrollTop = Math.clamp(body.scrollTop + speed, 0, body_inner.clientHeight - body.clientHeight + 3);
|
|
|
|
if (lef < 0) body.scrollLeft = body.scrollLeft - speed;
|
|
|
|
if (rig < 0) body.scrollLeft = Math.clamp(body.scrollLeft + speed, 0, body_inner.clientWidth - body.clientWidth);
|
2020-09-27 16:01:39 +08:00
|
|
|
|
|
|
|
updateKeyframeSelection()
|
2020-09-09 00:40:22 +08:00
|
|
|
},
|
|
|
|
end(e) {
|
|
|
|
if (!Timeline.selector.selecting) return false;
|
|
|
|
e.stopPropagation();
|
|
|
|
document.removeEventListener('mousemove', Timeline.selector.move);
|
|
|
|
clearInterval(Timeline.selector.interval);
|
|
|
|
document.removeEventListener('mouseup', Timeline.selector.end);
|
|
|
|
|
|
|
|
updateKeyframeSelection()
|
|
|
|
Timeline.selector.selected_before.empty();
|
|
|
|
Timeline.selector.selecting = false;
|
|
|
|
$('#timeline_selector')
|
|
|
|
.css('width', 0)
|
|
|
|
.css('height', 0)
|
|
|
|
.hide()
|
|
|
|
},
|
|
|
|
},
|
|
|
|
setTime(seconds, editing) {
|
|
|
|
seconds = limitNumber(seconds, 0, 1000)
|
|
|
|
Timeline.vue._data.playhead = seconds
|
|
|
|
Timeline.time = seconds
|
|
|
|
if (!editing) {
|
|
|
|
Timeline.setTimecode(seconds)
|
|
|
|
}
|
|
|
|
if (Timeline.getMaxLength() !== Timeline.vue._data.length) {
|
|
|
|
Timeline.updateSize()
|
|
|
|
}
|
|
|
|
Timeline.revealTime(seconds)
|
|
|
|
},
|
|
|
|
revealTime(time) {
|
|
|
|
var scroll = $('#timeline_body').scrollLeft()
|
|
|
|
var playhead = time * Timeline.vue._data.size + 8
|
|
|
|
if (playhead < scroll || playhead > scroll + $('#timeline_body').width() - Timeline.vue._data.head_width) {
|
|
|
|
$('#timeline_body').scrollLeft(playhead-16)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
setTimecode(time) {
|
|
|
|
let m = Math.floor(time/60)
|
|
|
|
let s = Math.floor(time%60)
|
|
|
|
let f = Math.floor((time%1) * 100)
|
|
|
|
if ((s+'').length === 1) {s = '0'+s}
|
|
|
|
if ((f+'').length === 1) {f = '0'+f}
|
|
|
|
$('#timeline_corner').text(m + ':' + s + ':' + f)
|
|
|
|
},
|
2020-09-14 18:44:43 +08:00
|
|
|
snapTime(time, animation) {
|
2020-09-09 00:40:22 +08:00
|
|
|
//return time;
|
|
|
|
if (time == undefined || isNaN(time)) {
|
|
|
|
time = Timeline.time;
|
|
|
|
}
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!animation) animation = Animation.selected;
|
2020-09-15 00:31:28 +08:00
|
|
|
var fps = Math.clamp(animation ? animation.snapping : settings.animation_snap.value, 1, 120);
|
2020-09-09 00:40:22 +08:00
|
|
|
return Math.clamp(Math.round(time*fps)/fps, 0);
|
|
|
|
},
|
|
|
|
getStep() {
|
2020-09-15 20:21:04 +08:00
|
|
|
return 1/Math.clamp(Animation.selected ? Animation.selected.snapping : settings.animation_snap.value, 1, 120);
|
2020-09-09 00:40:22 +08:00
|
|
|
},
|
|
|
|
setup() {
|
|
|
|
$('#timeline_body').mousedown(Timeline.selector.down)
|
|
|
|
|
|
|
|
$('#timeline_time').on('mousedown touchstart', e => {
|
|
|
|
if (e.which !== 1 && !event.changedTouches) return;
|
|
|
|
if (e.target.classList.contains('timeline_marker')) return;
|
|
|
|
|
|
|
|
if (e.target.id == 'timeline_endbracket') {
|
|
|
|
|
2020-09-15 20:21:04 +08:00
|
|
|
if (Animation.selected) {
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.dragging_endbracket = true;
|
2020-09-15 20:21:04 +08:00
|
|
|
Undo.initEdit({animations: [Animation.selected]});
|
2020-09-09 00:40:22 +08:00
|
|
|
} else {
|
|
|
|
Blockbench.showQuickMessage('message.no_animation_selected');
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
convertTouchEvent(e);
|
|
|
|
Timeline.dragging_playhead = true;
|
|
|
|
|
|
|
|
let offset = e.clientX - $('#timeline_time').offset().left;
|
|
|
|
let time = Timeline.snapTime(offset / Timeline.vue._data.size);
|
|
|
|
Timeline.setTime(time);
|
|
|
|
Animator.preview();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
$(document).on('mousemove touchmove', e => {
|
|
|
|
if (Timeline.dragging_playhead) {
|
|
|
|
|
|
|
|
convertTouchEvent(e);
|
|
|
|
let offset = e.clientX - $('#timeline_time').offset().left;
|
|
|
|
let time = Timeline.snapTime(offset / Timeline.vue._data.size)
|
|
|
|
if (Timeline.time != time) {
|
|
|
|
Timeline.setTime(time)
|
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
} else if (Timeline.dragging_endbracket) {
|
|
|
|
|
|
|
|
convertTouchEvent(e);
|
|
|
|
let offset = e.clientX - $('#timeline_time').offset().left;
|
|
|
|
let time = Timeline.snapTime(offset / Timeline.vue._data.size)
|
|
|
|
|
2020-09-15 20:21:04 +08:00
|
|
|
Animation.selected.setLength(time)
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.revealTime(time)
|
|
|
|
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.on('mouseup touchend', e => {
|
|
|
|
if (Timeline.dragging_playhead) {
|
|
|
|
delete Timeline.dragging_playhead
|
|
|
|
Timeline.pause();
|
|
|
|
|
|
|
|
} else if (Timeline.dragging_endbracket) {
|
|
|
|
|
|
|
|
Undo.finishEdit('Change Animation Length')
|
|
|
|
delete Timeline.dragging_endbracket
|
|
|
|
}
|
|
|
|
})
|
|
|
|
//Keyframe inputs
|
2020-09-14 00:31:46 +08:00
|
|
|
|
|
|
|
document.addEventListener('focus', event => {
|
|
|
|
if (event.target && event.target.parentElement && event.target.parentElement.classList.contains('keyframe_input')) {
|
2020-09-27 05:29:28 +08:00
|
|
|
Undo.initEdit({keyframes: Timeline.selected.slice()})
|
2020-09-14 00:31:46 +08:00
|
|
|
}
|
|
|
|
}, true)
|
|
|
|
document.addEventListener('focusout', event => {
|
|
|
|
if (event.target && event.target.parentElement && event.target.parentElement.classList.contains('keyframe_input')) {
|
|
|
|
Undo.finishEdit('edit keyframe')
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
})
|
2020-09-14 00:31:46 +08:00
|
|
|
|
2020-09-09 00:40:22 +08:00
|
|
|
//Enter Time
|
|
|
|
$('#timeline_corner').click(e => {
|
|
|
|
if ($('#timeline_corner').attr('contenteditable') == 'true') return;
|
|
|
|
|
|
|
|
$('#timeline_corner').attr('contenteditable', true).focus().select()
|
|
|
|
var times = $('#timeline_corner').text().split(':')
|
|
|
|
while (times.length < 3) {
|
|
|
|
times.push('00')
|
|
|
|
}
|
|
|
|
var node = $('#timeline_corner').get(0).childNodes[0]
|
2020-10-05 05:44:06 +08:00
|
|
|
var selection = window.getSelection();
|
2020-09-09 00:40:22 +08:00
|
|
|
var range = document.createRange();
|
|
|
|
|
|
|
|
var sel = [0, node.length]
|
|
|
|
if (e.offsetX < 24) {
|
|
|
|
sel = [0, times[0].length]
|
|
|
|
} else if (e.offsetX < 54) {
|
|
|
|
sel = [times[0].length+1, times[1].length]
|
|
|
|
} else if (e.offsetX < 80) {
|
|
|
|
sel = [times[0].length+times[1].length+2, times[2].length]
|
|
|
|
}
|
|
|
|
sel[1] = limitNumber(sel[0]+sel[1], sel[0], node.length)
|
|
|
|
|
|
|
|
range.setStart(node, sel[0])
|
|
|
|
range.setEnd(node, sel[1])
|
|
|
|
selection.removeAllRanges();
|
|
|
|
selection.addRange(range);
|
|
|
|
})
|
|
|
|
.on('focusout keydown', e => {
|
|
|
|
if (e.type === 'focusout' || Keybinds.extra.confirm.keybind.isTriggered(e) || Keybinds.extra.cancel.keybind.isTriggered(e)) {
|
|
|
|
$('#timeline_corner').attr('contenteditable', false)
|
|
|
|
Timeline.setTimecode(Timeline.time)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.on('keyup', e => {
|
|
|
|
var times = $('#timeline_corner').text().split(':')
|
|
|
|
times.forEach((t, i) => {
|
|
|
|
times[i] = parseInt(t)
|
|
|
|
if (isNaN(times[i])) {
|
|
|
|
times[i] = 0
|
|
|
|
}
|
|
|
|
})
|
|
|
|
while (times.length < 3) {
|
|
|
|
times.push(0)
|
|
|
|
}
|
|
|
|
var seconds
|
|
|
|
= times[0]*60
|
|
|
|
+ limitNumber(times[1], 0, 59)
|
|
|
|
+ limitNumber(times[2]/100, 0, 99)
|
|
|
|
if (Math.abs(seconds-Timeline.time) > 1e-3 ) {
|
|
|
|
Timeline.setTime(seconds, true)
|
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
$('#timeline_vue').on('mousewheel scroll', function(e) {
|
|
|
|
e.preventDefault()
|
2020-09-27 23:03:28 +08:00
|
|
|
let event = e.originalEvent;
|
|
|
|
let body = $('#timeline_body').get(0)
|
2020-09-09 00:40:22 +08:00
|
|
|
if (event.shiftKey) {
|
|
|
|
body.scrollLeft += event.deltaY/4
|
|
|
|
|
|
|
|
} else if (event.ctrlOrCmd) {
|
|
|
|
|
2020-09-27 23:03:28 +08:00
|
|
|
let offset = $('#timeline_body_inner').offset()
|
|
|
|
let offsetX = event.clientX - offset.left - Timeline.vue._data.head_width;
|
|
|
|
|
2020-09-09 00:40:22 +08:00
|
|
|
var zoom = 1 - event.deltaY/600
|
|
|
|
let original_size = Timeline.vue._data.size
|
2020-09-27 23:03:28 +08:00
|
|
|
let updated_size = limitNumber(Timeline.vue._data.size * zoom, 10, 1000)
|
|
|
|
Timeline.vue._data.size = updated_size;
|
|
|
|
|
|
|
|
body.scrollLeft += (updated_size - original_size) * (offsetX / original_size)
|
2020-09-09 00:40:22 +08:00
|
|
|
|
|
|
|
} else {
|
|
|
|
body.scrollTop += event.deltaY/6.25
|
|
|
|
}
|
|
|
|
Timeline.updateSize()
|
|
|
|
event.preventDefault();
|
|
|
|
});
|
|
|
|
$('#timeline_body').on('scroll', e => {
|
|
|
|
Timeline.vue._data.scroll_left = $('#timeline_body').scrollLeft()||0;
|
|
|
|
})
|
|
|
|
|
|
|
|
BarItems.slider_animation_speed.update()
|
|
|
|
Timeline.is_setup = true
|
|
|
|
Timeline.setTime(0)
|
|
|
|
},
|
|
|
|
update() {
|
|
|
|
//Draggable
|
|
|
|
$('#timeline_body .keyframe:not(.ui-draggable)').draggable({
|
|
|
|
axis: 'x',
|
|
|
|
distance: 4,
|
|
|
|
helper: () => $('<div></div>'),
|
|
|
|
start: function(event, ui) {
|
|
|
|
|
|
|
|
var id = $(event.target).attr('id');
|
|
|
|
var clicked = Timeline.keyframes.findInArray('uuid', id)
|
|
|
|
|
|
|
|
if (!$(event.target).hasClass('selected') && !event.shiftKey && Timeline.selected.length != 0) {
|
|
|
|
clicked.select()
|
|
|
|
} else if (clicked && !clicked.selected) {
|
|
|
|
clicked.select({shiftKey: true})
|
|
|
|
}
|
|
|
|
|
|
|
|
Undo.initEdit({keyframes: Timeline.selected})
|
|
|
|
Timeline.dragging_keyframes = true;
|
2020-10-24 06:15:38 +08:00
|
|
|
Timeline.dragging_range = [Infinity, 0];
|
2020-09-09 00:40:22 +08:00
|
|
|
|
|
|
|
for (var kf of Timeline.selected) {
|
2020-10-24 06:15:38 +08:00
|
|
|
kf.time_before = kf.time;
|
|
|
|
Timeline.dragging_range[0] = Math.min(Timeline.dragging_range[0], kf.time);
|
|
|
|
Timeline.dragging_range[1] = Math.max(Timeline.dragging_range[1], kf.time);
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
drag: function(event, ui) {
|
2020-10-24 06:15:38 +08:00
|
|
|
var difference = Math.clamp((ui.position.left - ui.originalPosition.left - 8) / Timeline.vue._data.size, -256, 256);
|
|
|
|
let [min, max] = Timeline.dragging_range;
|
2020-10-25 06:36:55 +08:00
|
|
|
let id = event.target?.id;
|
|
|
|
let target = Timeline.selected.find(kf => kf.uuid == id);
|
|
|
|
if (event.ctrlKey) {
|
|
|
|
var time_factor = (target && target.time_before < (min + max) / 2)
|
|
|
|
? ((max-min-difference) / (max-min))
|
|
|
|
: ((max-min+difference) / (max-min));
|
|
|
|
time_factor = Math.roundTo(time_factor, 2);
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
|
|
|
|
for (var kf of Timeline.selected) {
|
2020-10-24 06:15:38 +08:00
|
|
|
if (event.ctrlKey) {
|
2020-10-25 06:36:55 +08:00
|
|
|
if (target && target.time_before < (min + max) / 2) {
|
|
|
|
var t = max - (kf.time_before - max) * -time_factor;
|
|
|
|
} else {
|
|
|
|
var t = min + (kf.time_before - min) * time_factor;
|
|
|
|
}
|
2020-10-24 06:15:38 +08:00
|
|
|
} else {
|
|
|
|
var t = kf.time_before + difference;
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
kf.time = Timeline.snapTime(t);
|
|
|
|
}
|
2020-10-25 06:36:55 +08:00
|
|
|
if (event.ctrlKey) {
|
|
|
|
Blockbench.setStatusBarText(Math.round(time_factor * 100) + '%');
|
|
|
|
} else {
|
|
|
|
Blockbench.setStatusBarText(trimFloatNumber(difference));
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
BarItems.slider_keyframe_time.update()
|
|
|
|
Animator.preview()
|
|
|
|
},
|
|
|
|
stop: function(event, ui) {
|
|
|
|
var deleted = []
|
|
|
|
for (var kf of Timeline.selected) {
|
|
|
|
delete kf.time_before;
|
|
|
|
kf.replaceOthers(deleted);
|
|
|
|
}
|
2020-10-24 06:15:38 +08:00
|
|
|
delete Timeline.dragging_range;
|
2020-10-25 06:36:55 +08:00
|
|
|
Blockbench.setStatusBarText();
|
2020-09-09 00:40:22 +08:00
|
|
|
Undo.addKeyframeCasualties(deleted);
|
|
|
|
Undo.finishEdit('drag keyframes')
|
|
|
|
setTimeout(() => {
|
|
|
|
Timeline.dragging_keyframes = false;
|
|
|
|
}, 20)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
|
|
|
getMaxLength() {
|
|
|
|
var max_length = ($('#timeline_body').width()-8) / Timeline.vue._data.size;
|
2020-09-28 18:55:45 +08:00
|
|
|
if (Animation.selected) max_length = Math.max(max_length, Animation.selected.length)
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.keyframes.forEach((kf) => {
|
|
|
|
max_length = Math.max(max_length, kf.time)
|
|
|
|
})
|
|
|
|
max_length = Math.max(max_length, Timeline.time) + 50/Timeline.vue._data.size
|
|
|
|
return max_length;
|
|
|
|
},
|
|
|
|
updateSize() {
|
|
|
|
let size = Timeline.vue._data.size
|
|
|
|
Timeline.vue._data.length = Timeline.getMaxLength()
|
|
|
|
Timeline.vue._data.timecodes.empty()
|
|
|
|
|
|
|
|
var step = 1
|
|
|
|
if (size < 1) {step = 1}
|
|
|
|
else if (size < 20) {step = 4}
|
|
|
|
else if (size < 40) {step = 2}
|
|
|
|
else if (size < 100) {step = 1}
|
|
|
|
else if (size < 256) {step = 0.5}
|
|
|
|
else if (size < 520) {step = 0.25}
|
|
|
|
else if (size < 660) {step = 0.2}
|
|
|
|
else if (size < 860) {step = 0.1}
|
|
|
|
else {step = 0.05}
|
|
|
|
|
|
|
|
|
|
|
|
if (step < 1) {
|
|
|
|
var FPS = Timeline.getStep();
|
|
|
|
step = Math.round(step/FPS) * FPS
|
|
|
|
//step = 1/Math.round(1/step)
|
|
|
|
}
|
|
|
|
|
|
|
|
let substeps = step / Timeline.getStep()
|
|
|
|
while (substeps > 8) {
|
|
|
|
substeps /= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
var i = 0;
|
|
|
|
while (i < Timeline.vue._data.length) {
|
|
|
|
Timeline.vue._data.timecodes.push({
|
|
|
|
time: i,
|
|
|
|
width: step,
|
2020-09-28 18:55:45 +08:00
|
|
|
substeps,
|
2020-09-09 00:40:22 +08:00
|
|
|
text: Math.round(i*100)/100
|
|
|
|
})
|
|
|
|
i += step;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
updateScroll(e) {
|
|
|
|
$('.channel_head').css('left', scroll_amount+'px')
|
|
|
|
$('#timeline_time').css('left', -scroll_amount+'px')
|
|
|
|
},
|
|
|
|
unselect(e) {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) return;
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.keyframes.forEach((kf) => {
|
|
|
|
if (kf.selected) {
|
|
|
|
Timeline.selected.remove(kf)
|
|
|
|
}
|
|
|
|
kf.selected = false
|
|
|
|
})
|
|
|
|
TickUpdates.keyframe_selection = true;
|
|
|
|
},
|
|
|
|
start() {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) return;
|
|
|
|
Animation.selected.getMaxLength()
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.pause()
|
|
|
|
Timeline.playing = true
|
|
|
|
BarItems.play_animation.setIcon('pause')
|
2020-09-16 22:13:01 +08:00
|
|
|
Timeline.last_frame_timecode = new Date().getMilliseconds();
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.interval = setInterval(Timeline.loop, 100/6)
|
2020-09-15 20:21:04 +08:00
|
|
|
if (Animation.selected.loop == 'hold' && Timeline.time >= (Animation.selected.length||1e3)) {
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.setTime(0)
|
|
|
|
}
|
2020-09-13 02:35:25 +08:00
|
|
|
if (Timeline.time > 0) {
|
|
|
|
Animator.animations.forEach(animation => {
|
|
|
|
if (animation.playing && animation.animators.effects) {
|
|
|
|
animation.animators.effects.startPreviousSounds();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.loop()
|
|
|
|
},
|
|
|
|
loop() {
|
|
|
|
Animator.preview()
|
2020-09-15 20:21:04 +08:00
|
|
|
if (Animation.selected && Timeline.time < (Animation.selected.length||1e3)) {
|
2020-09-12 06:08:01 +08:00
|
|
|
|
2020-10-14 01:56:55 +08:00
|
|
|
var new_time;
|
|
|
|
if (Animation.selected?.anim_time_update) {
|
|
|
|
var new_time = Animator.MolangParser.parse(Animation.selected.anim_time_update);
|
|
|
|
}
|
|
|
|
if (new_time == undefined || new_time <= Timeline.time) {
|
|
|
|
var new_time = Animator.MolangParser.parse('query.anim_time + query.delta_time')
|
|
|
|
}
|
2020-09-12 20:59:49 +08:00
|
|
|
Timeline.setTime(Timeline.time + (new_time - Timeline.time) * (Timeline.playback_speed/100));
|
2020-09-16 22:13:01 +08:00
|
|
|
Timeline.last_frame_timecode = new Date().getMilliseconds();
|
2020-09-12 06:08:01 +08:00
|
|
|
|
2020-09-09 00:40:22 +08:00
|
|
|
} else {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (Animation.selected.loop == 'once') {
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.setTime(0)
|
|
|
|
Animator.preview()
|
|
|
|
Timeline.pause()
|
2020-09-15 20:21:04 +08:00
|
|
|
} else if (Animation.selected.loop == 'hold') {
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.pause()
|
|
|
|
} else {
|
|
|
|
Timeline.setTime(0)
|
|
|
|
Timeline.start()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
pause() {
|
2020-10-10 20:31:41 +08:00
|
|
|
Animator.preview();
|
2020-09-09 00:40:22 +08:00
|
|
|
Timeline.playing = false;
|
|
|
|
BarItems.play_animation.setIcon('play_arrow')
|
|
|
|
if (Timeline.interval) {
|
|
|
|
clearInterval(Timeline.interval)
|
|
|
|
Timeline.interval = false
|
|
|
|
}
|
|
|
|
Timeline.playing_sounds.forEach(media => {
|
|
|
|
if (!media.paused) {
|
|
|
|
media.pause();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
Timeline.playing_sounds.empty();
|
|
|
|
},
|
2020-09-13 02:35:25 +08:00
|
|
|
|
|
|
|
waveforms: {},
|
|
|
|
waveform_sample_rate: 60,
|
|
|
|
async visualizeAudioFile(path) {
|
|
|
|
|
|
|
|
if (!Timeline.waveforms[path]) {
|
|
|
|
Timeline.waveforms[path] = {
|
|
|
|
samples: [],
|
|
|
|
duration: 0
|
|
|
|
};
|
|
|
|
}
|
|
|
|
let {samples} = Timeline.waveforms[path];
|
|
|
|
|
|
|
|
let audioContext = new AudioContext()
|
|
|
|
let response = await fetch(path);
|
|
|
|
let arrayBuffer = await response.arrayBuffer();
|
|
|
|
let audioBuffer = await audioContext.decodeAudioData(arrayBuffer);
|
|
|
|
let data_array = audioBuffer.getChannelData(0);
|
|
|
|
|
|
|
|
Timeline.waveforms[path].duration = audioBuffer.duration;
|
|
|
|
|
|
|
|
// Sample
|
|
|
|
let sample_count = Math.ceil(audioBuffer.duration * Timeline.waveform_sample_rate);
|
|
|
|
samples.splice(0, samples.length);
|
|
|
|
for (var i = 0; i < sample_count; i++) {
|
|
|
|
samples.push(0);
|
|
|
|
}
|
|
|
|
for (var i = 0; i < data_array.length; i++) {
|
|
|
|
let sample_index = Math.floor((i / data_array.length) * sample_count);
|
|
|
|
samples[sample_index] += Math.abs(data_array[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Normalize
|
|
|
|
let max = Math.max(...samples);
|
|
|
|
samples.forEach((v, i) => samples[i] = v / max);
|
|
|
|
|
|
|
|
Timeline.vue.$forceUpdate();
|
|
|
|
|
|
|
|
return samples;
|
|
|
|
},
|
2020-10-09 20:10:48 +08:00
|
|
|
|
2020-09-09 00:40:22 +08:00
|
|
|
get keyframes() {
|
|
|
|
var keyframes = [];
|
|
|
|
Timeline.animators.forEach(animator => {
|
|
|
|
keyframes = [...keyframes, ...animator.keyframes]
|
|
|
|
})
|
|
|
|
return keyframes;
|
|
|
|
},
|
|
|
|
showMenu(event) {
|
|
|
|
if (event.target.nodeName == 'KEYFRAME' || event.target.parentElement.nodeName == 'KEYFRAME') return;
|
|
|
|
Timeline.menu.open(event, event);
|
|
|
|
},
|
|
|
|
menu: new Menu([
|
|
|
|
'paste',
|
|
|
|
'_',
|
|
|
|
'select_all',
|
|
|
|
'bring_up_all_animations',
|
|
|
|
'fold_all_animations',
|
|
|
|
'clear_timeline',
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
onVueSetup(function() {
|
|
|
|
Timeline.vue = new Vue({
|
|
|
|
el: '#timeline_vue',
|
|
|
|
data: {
|
|
|
|
size: 150,
|
|
|
|
length: 10,
|
|
|
|
animation_length: 0,
|
|
|
|
scroll_left: 0,
|
|
|
|
head_width: 180,
|
|
|
|
timecodes: [],
|
|
|
|
animators: Timeline.animators,
|
|
|
|
markers: [],
|
2020-09-13 02:35:25 +08:00
|
|
|
waveforms: Timeline.waveforms,
|
2020-09-09 00:40:22 +08:00
|
|
|
focus_channel: null,
|
2020-09-28 18:55:45 +08:00
|
|
|
playhead: Timeline.time,
|
|
|
|
channels: {
|
|
|
|
rotation: true,
|
|
|
|
position: true,
|
|
|
|
scale: true,
|
|
|
|
hide_empty: false,
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
toggleAnimator(animator) {
|
|
|
|
animator.expanded = !animator.expanded;
|
|
|
|
},
|
|
|
|
removeAnimator(animator) {
|
|
|
|
Timeline.animators.remove(animator);
|
2020-09-12 22:55:32 +08:00
|
|
|
},
|
|
|
|
getColor(index) {
|
|
|
|
if (index == -1 || index == undefined) return;
|
|
|
|
return markerColors[index].standard;
|
2020-09-13 02:35:25 +08:00
|
|
|
},
|
|
|
|
getWaveformPoints(samples, size) {
|
|
|
|
let height = 23;
|
|
|
|
let points = [`0,${height}`];
|
|
|
|
samples.forEach((sample, i) => {
|
|
|
|
points.push(`${(i + 0.5) / Timeline.waveform_sample_rate * size},${(1 - sample) * height}`);
|
|
|
|
})
|
|
|
|
points.push(`${(samples.length) / Timeline.waveform_sample_rate * size},${height}`)
|
|
|
|
return points.join(' ');
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
BARS.defineActions(function() {
|
|
|
|
new Action('play_animation', {
|
|
|
|
icon: 'play_arrow',
|
|
|
|
category: 'animation',
|
|
|
|
keybind: new Keybind({key: 32}),
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function () {
|
|
|
|
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) {
|
2020-09-09 00:40:22 +08:00
|
|
|
Blockbench.showQuickMessage('message.no_animation_selected')
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (Timeline.playing) {
|
|
|
|
Timeline.pause()
|
|
|
|
} else {
|
|
|
|
Timeline.start()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new NumSlider('slider_animation_speed', {
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
get: function() {
|
|
|
|
return Timeline.playback_speed;
|
|
|
|
},
|
|
|
|
change: function(modify) {
|
|
|
|
Timeline.playback_speed = limitNumber(modify(Timeline.playback_speed), 0, 10000)
|
|
|
|
},
|
|
|
|
getInterval: (e) => {
|
|
|
|
var val = BarItems.slider_animation_speed.get()
|
|
|
|
if (e.shiftKey) {
|
|
|
|
if (val < 50) {
|
|
|
|
return 10;
|
|
|
|
} else {
|
|
|
|
return 50;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (e.ctrlOrCmd) {
|
|
|
|
if (val < 500) {
|
|
|
|
return 1;
|
|
|
|
} else {
|
|
|
|
return 10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (val < 10) {
|
|
|
|
return 1;
|
|
|
|
} else if (val < 50) {
|
|
|
|
return 5;
|
|
|
|
} else if (val < 160) {
|
|
|
|
return 10;
|
|
|
|
} else if (val < 300) {
|
|
|
|
return 20;
|
|
|
|
} else if (val < 1000) {
|
|
|
|
return 50;
|
|
|
|
} else {
|
|
|
|
return 500;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('jump_to_timeline_start', {
|
|
|
|
icon: 'skip_previous',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
keybind: new Keybind({key: 36}),
|
|
|
|
click: function () {
|
|
|
|
Timeline.setTime(0)
|
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
new Action('jump_to_timeline_end', {
|
|
|
|
icon: 'skip_next',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
keybind: new Keybind({key: 35}),
|
|
|
|
click: function () {
|
2020-09-15 20:21:04 +08:00
|
|
|
Timeline.setTime(Animation.selected ? Animation.selected.length : 0)
|
2020-09-09 00:40:22 +08:00
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
})
|
2020-10-08 04:15:29 +08:00
|
|
|
new Action('timeline_frame_back', {
|
|
|
|
icon: 'arrow_back',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
keybind: new Keybind({key: 188}),
|
|
|
|
click: function (e) {
|
|
|
|
let time = Timeline.snapTime(limitNumber(Timeline.time - Timeline.getStep(), 0, 1e4));
|
|
|
|
Timeline.setTime(time);
|
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('timeline_frame_forth', {
|
|
|
|
icon: 'arrow_forward',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
keybind: new Keybind({key: 190}),
|
|
|
|
click: function (e) {
|
|
|
|
let time = Timeline.snapTime(limitNumber(Timeline.time + Timeline.getStep(), 0, 1e4));
|
|
|
|
Timeline.setTime(time);
|
|
|
|
Animator.preview()
|
|
|
|
}
|
|
|
|
})
|
2020-09-09 00:40:22 +08:00
|
|
|
|
|
|
|
new Action('bring_up_all_animations', {
|
|
|
|
icon: 'fa-sort-amount-up',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function () {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) return;
|
|
|
|
for (var uuid in Animation.selected.animators) {
|
|
|
|
var ba = Animation.selected.animators[uuid]
|
2020-10-05 05:44:06 +08:00
|
|
|
if (ba && ba.keyframes.length) {
|
|
|
|
ba.addToTimeline();
|
|
|
|
}
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('fold_all_animations', {
|
|
|
|
icon: 'format_indent_decrease',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function () {
|
|
|
|
for (var animator of Timeline.animators) {
|
2020-10-05 05:44:06 +08:00
|
|
|
animator.expanded = false;
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('clear_timeline', {
|
|
|
|
icon: 'clear_all',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function () {
|
|
|
|
Timeline.vue._data.animators.purge();
|
|
|
|
unselectAll();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('select_effect_animator', {
|
|
|
|
icon: 'fa-magic',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function () {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) return;
|
|
|
|
if (!Animation.selected.animators.effects) {
|
|
|
|
var ea = Animation.selected.animators.effects = new EffectAnimator(Animation.selected);
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
2020-09-15 20:21:04 +08:00
|
|
|
Animation.selected.animators.effects.select()
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
})
|
2020-09-28 18:55:45 +08:00
|
|
|
new Action('timeline_focus', {
|
|
|
|
icon: 'fas.fa-filter',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
click: function (e) {
|
|
|
|
new Menu(this.children()).open(e.target)
|
2020-09-09 00:40:22 +08:00
|
|
|
},
|
2020-09-28 18:55:45 +08:00
|
|
|
children: function() {
|
|
|
|
let on = 'fas.fa-check-square';
|
|
|
|
let off = 'far.fa-square';
|
|
|
|
let {channels} = Timeline.vue._data;
|
|
|
|
return [
|
|
|
|
{name: 'timeline.rotation', icon: channels.rotation ? on : off, click() {channels.rotation = !channels.rotation}},
|
|
|
|
{name: 'timeline.position', icon: channels.position ? on : off, click() {channels.position = !channels.position}},
|
|
|
|
{name: 'timeline.scale', icon: channels.scale ? on : off, click() {channels.scale = !channels.scale}},
|
2020-09-28 23:26:39 +08:00
|
|
|
'_',
|
2020-09-28 18:55:45 +08:00
|
|
|
{name: 'action.timeline_focus.hide_empty', icon: channels.hide_empty ? on : off, click() {channels.hide_empty = !channels.hide_empty}},
|
|
|
|
]
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
})
|
|
|
|
new Action('add_marker', {
|
|
|
|
icon: 'flag',
|
|
|
|
category: 'animation',
|
|
|
|
condition: {modes: ['animate']},
|
|
|
|
keybind: new Keybind({ctrl: true, key: 77}),
|
|
|
|
click: function (event) {
|
2020-09-15 20:21:04 +08:00
|
|
|
if (!Animation.selected) {
|
2020-09-09 00:40:22 +08:00
|
|
|
Blockbench.showQuickMessage('message.no_animation_selected')
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var time = Timeline.snapTime();
|
|
|
|
var original_marker;
|
2020-09-15 20:21:04 +08:00
|
|
|
for (var m of Animation.selected.markers) {
|
2020-09-09 00:40:22 +08:00
|
|
|
if (Math.abs(m.time - time) < 0.01) {
|
|
|
|
original_marker = m;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (original_marker) {
|
2020-09-15 20:21:04 +08:00
|
|
|
Animation.selected.markers.remove(original_marker);
|
2020-09-09 00:40:22 +08:00
|
|
|
} else {
|
|
|
|
let marker = new TimelineMarker({time});
|
2020-09-15 20:21:04 +08:00
|
|
|
Animation.selected.markers.push(marker);
|
2020-09-09 00:40:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|