blockbench/js/interface/interface.js

864 lines
26 KiB
JavaScript
Raw Normal View History

2019-07-18 00:02:07 +08:00
class ResizeLine {
2022-02-23 01:12:27 +08:00
constructor(id, data) {
2019-07-18 00:02:07 +08:00
var scope = this;
2022-02-23 01:12:27 +08:00
if (typeof id == 'object') {
data = id;
id = data.id;
}
this.id = id
2019-07-18 00:02:07 +08:00
this.horizontal = data.horizontal === true
this.position = data.position
this.condition = data.condition
this.width = 0;
2022-02-23 01:12:27 +08:00
this.get = data.get;
this.set = data.set;
this.node = document.createElement('div');
this.node.className = 'resizer '+(data.horizontal ? 'horizontal' : 'vertical');
this.node.id = 'resizer_'+this.id;
$(this.node).draggable({
2020-04-26 02:25:07 +08:00
axis: this.horizontal ? 'y' : 'x',
containment: '#work_screen',
2019-07-18 00:02:07 +08:00
revert: true,
2020-04-26 02:25:07 +08:00
revertDuration: 0,
2022-02-23 01:12:27 +08:00
start(e, u) {
2019-07-18 00:02:07 +08:00
scope.before = data.get()
},
2022-02-23 01:12:27 +08:00
drag(e, u) {
2019-07-18 00:02:07 +08:00
if (scope.horizontal) {
data.set(scope.before, u.position.top - u.originalPosition.top)
} else {
2020-04-26 02:25:07 +08:00
data.set(scope.before, (u.position.left - u.originalPosition.left))
2019-07-18 00:02:07 +08:00
}
updateInterface()
},
2022-02-23 01:12:27 +08:00
stop(e, u) {
2019-07-18 00:02:07 +08:00
updateInterface()
scope.update();
2019-07-18 00:02:07 +08:00
}
})
}
update() {
if (BARS.condition(this.condition)) {
$(this.node).show()
if (this.position) {
2022-02-23 01:12:27 +08:00
this.position.call(this, this)
2019-07-18 00:02:07 +08:00
}
} else {
$(this.node).hide()
}
}
setPosition(data) {
var jq = $(this.node)
jq.css('top', data.top !== undefined ? data.top+ 'px' : '')
jq.css('bottom',data.bottom !== undefined ? data.bottom+'px' : '')
jq.css('left', data.left !== undefined ? data.left+ 'px' : '')
jq.css('right', data.right !== undefined ? data.right+ 'px' : '')
if (data.top !== undefined) {
jq.css('top', data.top+'px')
}
2022-07-27 00:44:56 +08:00
if (data.bottom !== undefined && (!this.horizontal || data.top === undefined)) {
2019-07-18 00:02:07 +08:00
jq.css('bottom', data.bottom+'px')
}
if (data.left !== undefined) {
jq.css('left', data.left+'px')
}
2022-07-27 00:44:56 +08:00
if (data.right !== undefined && (this.horizontal || data.left === undefined)) {
2019-07-18 00:02:07 +08:00
jq.css('right', data.right+'px')
}
}
}
2020-07-16 15:32:59 +08:00
const Interface = {
2019-07-18 00:02:07 +08:00
default_data: {
left_bar_width: 366,
2019-07-26 19:33:29 +08:00
right_bar_width: 314,
2019-07-18 00:02:07 +08:00
quad_view_x: 50,
quad_view_y: 50,
2021-11-13 23:34:47 +08:00
timeline_head: Blockbench.isMobile ? 140 : 196,
2022-08-06 19:37:20 +08:00
left_bar: ['uv', 'color', 'textures', 'display', 'animations', 'keyframe', 'variable_placeholders'],
2022-03-04 04:13:30 +08:00
right_bar: ['element', 'bone', 'color', 'skin_pose', 'outliner', 'chat'],
panels: {
paint: {
slot: 'left_bar',
float_position: [300, 0],
float_size: [500, 600],
height: window.innerHeight/2-50
2022-08-06 19:37:20 +08:00
},
color_2d: {
slot: 'left_bar',
float_position: [50, 0],
float_size: [300, 400],
height: 400
}
}
2019-07-18 00:02:07 +08:00
},
2021-01-10 01:33:42 +08:00
get left_bar_width() {
return Prop.show_left_bar && Interface.getLeftPanels().length ? Interface.data.left_bar_width : 0;
2021-01-10 01:33:42 +08:00
},
get right_bar_width() {
return Prop.show_right_bar && Interface.getRightPanels().length ? Interface.data.right_bar_width : 0;
2021-01-10 01:33:42 +08:00
},
2022-02-23 01:12:27 +08:00
get top_panel_height() {
return 1;
},
get bottom_panel_height() {
return 1;
},
getTopPanel() {
for (let key in Panels) {
let panel = Panels[key];
if (panel.slot == 'top' && Condition(panel.condition)) {
return panel;
}
}
},
getBottomPanel() {
for (let key in Panels) {
let panel = Panels[key];
if (panel.slot == 'bottom' && Condition(panel.condition)) {
return panel;
}
}
},
getLeftPanels() {
let list = [];
for (let key in Panels) {
let panel = Panels[key];
if (panel.slot == 'left_bar' && Condition(panel.condition)) {
list.push(panel);
}
}
return list;
},
getRightPanels() {
let list = [];
for (let key in Panels) {
let panel = Panels[key];
if (panel.slot == 'right_bar' && Condition(panel.condition)) {
list.push(panel);
}
}
return list;
},
2019-07-18 00:02:07 +08:00
Resizers: {
2022-02-23 01:12:27 +08:00
left: new ResizeLine('left', {
condition() {
if (Blockbench.isMobile) return false;
2021-01-10 01:33:42 +08:00
if (!Prop.show_left_bar) return false;
2020-07-16 15:32:59 +08:00
for (let p of Interface.data.left_bar) {
if (Panels[p] && BARS.condition(Panels[p].condition) && Panels[p].slot == 'left_bar') {
2020-07-16 15:32:59 +08:00
return true;
}
}
2019-07-18 00:02:07 +08:00
},
2022-02-23 01:12:27 +08:00
get() {return Interface.data.left_bar_width},
set(o, diff) {
2021-04-11 19:15:21 +08:00
let min = 128;
let calculated = limitNumber(o + diff, min, window.innerWidth- 120 - Interface.data.right_bar_width)
2020-04-26 02:25:07 +08:00
Interface.data.left_bar_width = Math.snapToValues(calculated, [Interface.default_data.left_bar_width], 16);
2021-04-11 19:15:21 +08:00
if (calculated == min) {
Prop.show_left_bar = false;
Interface.data.left_bar_width = Interface.default_data.left_bar_width;
} else {
Prop.show_left_bar = true;
}
2019-07-18 00:02:07 +08:00
},
2022-02-23 01:12:27 +08:00
position() {
this.setPosition({
top: 0,
2019-07-18 00:02:07 +08:00
bottom: 0,
2020-04-26 02:25:07 +08:00
left: Interface.data.left_bar_width+2
2019-07-18 00:02:07 +08:00
})
}
}),
2022-02-23 01:12:27 +08:00
right: new ResizeLine('right', {
condition() {
if (Blockbench.isMobile) return false;
2021-01-10 01:33:42 +08:00
if (!Prop.show_right_bar) return false;
2020-07-16 15:32:59 +08:00
for (let p of Interface.data.right_bar) {
if (Panels[p] && BARS.condition(Panels[p].condition) && Panels[p].slot == 'right_bar') {
2020-07-16 15:32:59 +08:00
return true;
}
}
2019-07-18 00:02:07 +08:00
},
2022-02-23 01:12:27 +08:00
get() {return Interface.data.right_bar_width},
set(o, diff) {
2021-04-11 19:15:21 +08:00
let min = 128;
let calculated = limitNumber(o - diff, min, window.innerWidth- 120 - Interface.data.left_bar_width);
2020-04-26 02:25:07 +08:00
Interface.data.right_bar_width = Math.snapToValues(calculated, [Interface.default_data.right_bar_width], 12);
2021-04-11 19:15:21 +08:00
if (calculated == min) {
Prop.show_right_bar = false;
Interface.data.right_bar_width = Interface.default_data.right_bar_width;
} else {
Prop.show_right_bar = true;
}
2019-07-18 00:02:07 +08:00
},
2022-02-23 01:12:27 +08:00
position() {
this.setPosition({
top: 30,
2019-07-18 00:02:07 +08:00
bottom: 0,
2020-04-26 02:25:07 +08:00
right: Interface.data.right_bar_width-2
2019-07-18 00:02:07 +08:00
})
}
}),
2022-02-23 01:12:27 +08:00
quad_view_x: new ResizeLine('quad_view_x', {
condition() {return quad_previews.enabled},
get() {return Interface.data.quad_view_x},
set(o, diff) {Interface.data.quad_view_x = limitNumber(o + diff/$('#preview').width()*100, 5, 95)},
position() {
2019-08-18 00:26:14 +08:00
var p = document.getElementById('preview')
2022-02-23 01:12:27 +08:00
this.setPosition({
2019-08-18 00:26:14 +08:00
top: 32,
bottom: p ? window.innerHeight - (p.clientHeight + $(p).offset().top) : 0,
2021-11-13 23:34:47 +08:00
left: Interface.left_bar_width + document.getElementById('preview').clientWidth*Interface.data.quad_view_x/100
2019-08-18 00:26:14 +08:00
}
)}
2019-07-18 00:02:07 +08:00
}),
2022-02-23 01:12:27 +08:00
quad_view_y: new ResizeLine('quad_view_y', {
2019-07-18 00:02:07 +08:00
horizontal: true,
2022-02-23 01:12:27 +08:00
condition() {return quad_previews.enabled},
get() {return Interface.data.quad_view_y},
set(o, diff) {
2021-11-13 23:34:47 +08:00
Interface.data.quad_view_y = limitNumber(o + diff/document.getElementById('preview').clientHeight*100, 5, 95)
2019-07-18 00:02:07 +08:00
},
2022-02-23 01:12:27 +08:00
position() {this.setPosition({
2021-01-10 01:33:42 +08:00
left: Interface.left_bar_width+2,
right: Interface.right_bar_width+2,
top: Interface.preview.offsetTop + 30 + Interface.preview.clientHeight*Interface.data.quad_view_y/100
2019-07-18 00:02:07 +08:00
})}
2019-08-18 00:26:14 +08:00
}),
2022-02-23 01:12:27 +08:00
top: new ResizeLine('top', {
2019-08-18 00:26:14 +08:00
horizontal: true,
condition() {return !Blockbench.isMobile && Interface.getTopPanel()},
2022-03-04 04:13:30 +08:00
get() {
let panel = Interface.getTopPanel();
2022-03-11 01:12:59 +08:00
return panel.folded ? panel.handle.clientHeight : panel.height;
2022-03-04 04:13:30 +08:00
},
2022-02-23 01:12:27 +08:00
set(o, diff) {
2022-03-04 04:13:30 +08:00
let panel = Interface.getTopPanel();
2022-03-11 01:12:59 +08:00
panel.position_data.height = limitNumber(o + diff, 150);
if (panel.folded) panel.fold(false);
2022-03-04 04:13:30 +08:00
panel.update();
2022-03-11 01:12:59 +08:00
if (Interface.getBottomPanel()) Interface.getBottomPanel().update();
2019-08-18 00:26:14 +08:00
},
2022-02-23 01:12:27 +08:00
position() {this.setPosition({
2021-01-10 01:33:42 +08:00
left: Interface.left_bar_width+2,
right: Interface.right_bar_width+2,
2022-03-05 06:34:36 +08:00
top: this.get() + Interface.work_screen.offsetTop - document.getElementById('page_wrapper').offsetTop
2019-08-18 00:26:14 +08:00
})}
2021-11-13 23:34:47 +08:00
}),
2022-02-23 01:12:27 +08:00
bottom: new ResizeLine('bottom', {
horizontal: true,
condition() {return !Blockbench.isMobile && Interface.getBottomPanel()},
2022-03-04 04:13:30 +08:00
get() {
let panel = Interface.getBottomPanel();
2022-03-11 01:12:59 +08:00
return panel.folded ? panel.handle.clientHeight : panel.height;
2022-03-04 04:13:30 +08:00
},
2022-02-23 01:12:27 +08:00
set(o, diff) {
let panel = Interface.getBottomPanel();
panel.position_data.height = limitNumber(o - diff, 150);
if (panel.folded) panel.fold(false);
panel.update();
2022-03-11 01:12:59 +08:00
if (Interface.getTopPanel()) Interface.getTopPanel().update();
2022-02-23 01:12:27 +08:00
},
position() {this.setPosition({
left: Interface.left_bar_width+2,
right: Interface.right_bar_width+2,
2022-03-05 06:34:36 +08:00
top: Interface.work_screen.clientHeight - document.getElementById('status_bar').clientHeight - this.get()
2022-02-23 01:12:27 +08:00
})}
}),
timeline_head: new ResizeLine('timeline_head', {
2021-11-13 23:34:47 +08:00
horizontal: false,
condition() {return Modes.animate},
get() {return Interface.data.timeline_head},
set(o, diff) {
let value = limitNumber(o + diff, 90, Panels.timeline.node.clientWidth - 40);
2021-11-13 23:34:47 +08:00
value = Math.snapToValues(value, [Interface.default_data.timeline_head], 12);
Interface.data.timeline_head = Timeline.vue._data.head_width = value;
},
2022-03-04 04:13:30 +08:00
position() {
let offset = $(Panels.timeline.vue.$el).offset();
2022-03-04 04:13:30 +08:00
this.setPosition({
left: offset.left + 2 + Interface.data.timeline_head,
top: offset.top - Interface.work_screen.offsetTop + 30,
bottom: Interface.work_screen.clientHeight - offset.top + Interface.work_screen.offsetTop - Panels.timeline.vue.$el.clientHeight + 10
2022-03-04 04:13:30 +08:00
})
}
2019-07-18 00:02:07 +08:00
})
},
CustomElements: {},
2019-07-18 00:02:07 +08:00
status_bar: {},
2021-01-10 01:33:42 +08:00
Panels: {},
2022-03-11 01:12:59 +08:00
toggleSidebar(side, status) {
if (status == undefined) status = !Prop[`show_${side}_bar`];
Prop[`show_${side}_bar`] = !!status;
2021-01-10 01:33:42 +08:00
resizeWindow();
}
2019-07-18 00:02:07 +08:00
}
2022-02-23 01:12:27 +08:00
const Panels = Interface.Panels;
2020-09-22 05:23:42 +08:00
Interface.panel_definers = []
Interface.definePanels = function(callback) {
Interface.panel_definers.push(callback);
2022-02-23 01:12:27 +08:00
};
2022-02-23 01:12:27 +08:00
(function() {
2019-07-18 00:02:07 +08:00
Interface.data = $.extend(true, {}, Interface.default_data)
var interface_data = localStorage.getItem('interface_data')
2022-03-04 04:13:30 +08:00
if (!interface_data) return;
2019-07-18 00:02:07 +08:00
try {
interface_data = JSON.parse(interface_data)
let original_left_bar, original_right_bar;
if (interface_data.left_bar) {
original_left_bar = Interface.data.left_bar;
Interface.data.left_bar = interface_data.left_bar;
}
if (interface_data.right_bar) {
original_right_bar = Interface.data.right_bar;
Interface.data.right_bar = interface_data.right_bar;
}
if (original_left_bar) {
original_left_bar.forEach((panel, i) => {
if (Interface.data.left_bar.includes(panel)) return;
if (Interface.data.right_bar.includes(panel)) return;
Interface.data.left_bar.splice(i, 0, panel);
})
}
if (original_right_bar) {
original_right_bar.forEach((panel, i) => {
if (Interface.data.right_bar.includes(panel)) return;
if (Interface.data.left_bar.includes(panel)) return;
Interface.data.right_bar.splice(i, 0, panel);
})
}
2019-07-18 00:02:07 +08:00
$.extend(true, Interface.data, interface_data)
2022-03-04 04:13:30 +08:00
} catch (err) {
console.error(err);
}
2022-02-23 01:12:27 +08:00
})()
//Misc
function unselectInterface(event) {
if (open_menu && $('.contextMenu').find(event.target).length === 0 && $('.menu_bar_point.opened:hover').length === 0) {
Menu.closed_in_this_click = open_menu.id;
open_menu.hide();
function mouseUp(e) {
delete Menu.closed_in_this_click;
document.removeEventListener('click', mouseUp);
}
document.addEventListener('click', mouseUp);
}
if (ActionControl.open && $('#action_selector').find(event.target).length === 0 && (!open_menu || open_menu instanceof BarMenu)) {
ActionControl.hide();
}
if ($(event.target).is('input.cube_name:not([disabled])') === false && Blockbench.hasFlag('renaming')) {
stopRenameOutliner()
}
}
function setupInterface() {
2019-07-18 00:02:07 +08:00
2021-02-10 05:26:52 +08:00
translateUI()
2021-11-22 02:28:19 +08:00
if (Blockbench.isMobile) document.body.classList.add('is_mobile');
if (Blockbench.isLandscape) document.body.classList.add('is_landscape');
if (Blockbench.isTouch) document.body.classList.add('is_touch');
2021-11-22 02:28:19 +08:00
document.getElementById('title_bar_home_button').title = tl('projects.start_screen');
2019-07-18 00:02:07 +08:00
document.getElementById('center').classList.toggle('checkerboard', settings.preview_checkerboard.value);
document.body.classList.toggle('mobile_sidebar_left', settings.mobile_panel_side.value == 'left');
2020-03-05 03:56:17 +08:00
2020-07-16 15:32:59 +08:00
setupPanels()
2022-02-23 01:12:27 +08:00
Interface.status_bar.menu = new Menu([
'project_window',
'open_model_folder',
'open_backup_folder',
'save',
'timelapse',
'cancel_gif',
2022-02-23 01:12:27 +08:00
])
if (Blockbench.isMobile) {
setupMobilePanelSelector()
Prop.show_right_bar = false;
Prop.show_left_bar = false;
}
2019-07-18 00:02:07 +08:00
for (var key in Interface.Resizers) {
var resizer = Interface.Resizers[key]
$('#work_screen').append(resizer.node)
2019-07-18 00:02:07 +08:00
}
//$(document).contextmenu()
if (Blockbench.isMobile) {
document.getElementById('preview').append(
document.querySelector('.toolbar_wrapper.narrow.tools')
);
}
2019-07-18 00:02:07 +08:00
//Tooltip Fix
2019-08-18 00:26:14 +08:00
$(document).on('mouseenter', '.tool', function() {
2019-07-18 00:02:07 +08:00
var tooltip = $(this).find('div.tooltip')
if (!tooltip || typeof tooltip.offset() !== 'object') return;
//Left
if (tooltip.css('left') === '-4px') {
tooltip.css('left', 'auto')
}
if (-tooltip.offset().left > 4) {
tooltip.css('left', '-4px')
}
//Right
if (tooltip.css('right') === '-4px') {
tooltip.css('right', 'auto')
}
2019-08-18 00:26:14 +08:00
if ((tooltip.offset().left + tooltip.width()) - window.innerWidth > 4) {
2019-07-18 00:02:07 +08:00
tooltip.css('right', '-4px')
2019-08-18 00:26:14 +08:00
} else if ($(this).parent().css('position') == 'relative') {
tooltip.css('right', '0')
2019-07-18 00:02:07 +08:00
}
})
//Clickbinds
2021-01-31 05:30:34 +08:00
$('#preview').click(function() { setActivePanel('preview' )})
2019-07-18 00:02:07 +08:00
$('#texture_list').click(function(){
unselectTextures()
})
2022-03-04 04:13:30 +08:00
$(Panels.timeline.node).mousedown((event) => {
2019-07-18 00:02:07 +08:00
setActivePanel('timeline');
})
2019-12-16 03:04:31 +08:00
$(document).on('mousedown touchstart', unselectInterface)
2021-03-11 02:25:56 +08:00
window.addEventListener('resize', resizeWindow);
window.addEventListener('orientationchange', () => {
setTimeout(resizeWindow, 100)
});
2021-08-19 01:12:34 +08:00
setProjectTitle()
2021-03-11 02:25:56 +08:00
Interface.text_edit_menu = new Menu([
{
id: 'copy',
name: 'action.copy',
2021-03-11 02:25:56 +08:00
icon: 'fa-copy',
click() {
document.execCommand('copy');
}
},
{
id: 'paste',
name: 'action.paste',
2021-03-11 02:25:56 +08:00
icon: 'fa-paste',
click() {
document.execCommand('paste');
}
}
])
document.oncontextmenu = function (event) {
if (!$(event.target).hasClass('allow_default_menu') && (!Blockbench.isTouch || event instanceof TouchEvent == false)) {
2021-03-11 02:25:56 +08:00
if (event.target.nodeName === 'INPUT' && $(event.target).is(':focus')) {
2019-07-18 00:02:07 +08:00
Interface.text_edit_menu.open(event, event.target)
2021-03-11 02:25:56 +08:00
}
2019-07-18 00:02:07 +08:00
return false;
}
}
2019-07-18 00:02:07 +08:00
//Scrolling
$('input[type="range"]').on('mousewheel', function () {
var obj = $(event.target)
var factor = event.deltaY > 0 ? -1 : 1
var val = parseFloat(obj.val()) + parseFloat(obj.attr('step')) * factor
val = limitNumber(val, obj.attr('min'), obj.attr('max'))
if (obj.attr('trigger_type')) {
DisplayMode.scrollSlider(obj.attr('trigger_type'), val, obj)
return;
}
obj.val(val)
eval(obj.attr('oninput'))
eval(obj.attr('onmouseup'))
})
//Mousemove
$(document).mousemove(function(event) {
mouse_pos.x = event.clientX
mouse_pos.y = event.clientY
})
updateInterface()
}
function updateInterface() {
BARS.updateConditions()
MenuBar.update()
updatePanelSelector();
2019-07-18 00:02:07 +08:00
resizeWindow()
localStorage.setItem('interface_data', JSON.stringify(Interface.data))
}
function resizeWindow(event) {
2021-03-11 02:25:56 +08:00
if (!Preview.all || (event && event.target && event.target !== window)) {
2019-07-18 00:02:07 +08:00
return;
}
Blockbench.isLandscape = window.innerWidth > window.innerHeight;
document.body.classList.toggle('is_landscape', Blockbench.isLandscape);
2019-07-18 00:02:07 +08:00
if (Interface.data) {
updateInterfacePanels()
}
2020-07-16 15:32:59 +08:00
Preview.all.forEach(function(prev) {
2019-07-18 00:02:07 +08:00
if (prev.canvas.isConnected) {
prev.resize()
}
})
Outliner.elements.forEach(element => {
if (element.preview_controller.updateWindowSize) {
element.preview_controller.updateWindowSize(element);
}
})
if (Format.image_editor) {
UVEditor.updateSize();
}
2019-07-18 00:02:07 +08:00
var dialog = $('dialog#'+open_dialog)
if (dialog.length) {
if (dialog.outerWidth() + dialog.offset().left > window.innerWidth) {
dialog.css('left', limitNumber(window.innerWidth-dialog.outerWidth(), 0, 4e3) + 'px')
}
if (dialog.outerHeight() + dialog.offset().top > window.innerHeight) {
dialog.css('top', limitNumber(window.innerHeight-dialog.outerHeight(), 0, 4e3) + 'px')
}
}
Blockbench.dispatchEvent('resize_window', event);
2019-07-18 00:02:07 +08:00
}
2019-07-18 00:02:07 +08:00
function setProjectTitle(title) {
2021-07-11 04:22:02 +08:00
let window_title = 'Blockbench';
if (title == undefined && Project.name) {
title = Project.name
2019-07-18 00:02:07 +08:00
}
if (title) {
Prop.file_name = Prop.file_name_alt = title
if (!Project.name) {
Project.name = title
}
if (Format.bone_rig) {
title = title.replace(/^geometry\./,'').replace(/:[a-z0-9.]+/, '')
}
2021-07-11 04:22:02 +08:00
window_title = title+' - Blockbench';
2019-07-18 00:02:07 +08:00
} else {
Prop.file_name = Prop.file_name_alt = ''
}
2021-11-22 02:28:19 +08:00
if (Project && !Project.saved) window_title = '● ' + window_title;
2021-07-11 04:22:02 +08:00
$('title').text(window_title);
$('#header_free_bar').text(window_title);
2019-07-18 00:02:07 +08:00
}
//Zoom
function setZoomLevel(mode) {
if (Prop.active_panel === 'uv') {
2021-08-18 04:02:23 +08:00
var zoom = UVEditor.zoom
2019-07-18 00:02:07 +08:00
switch (mode) {
case 'in': zoom *= 1.5; break;
case 'out': zoom *= 0.66; break;
case 'reset': zoom = 1; break;
}
UVEditor.setZoom(zoom);
2019-07-18 00:02:07 +08:00
} else if (Prop.active_panel == 'timeline') {
let body = document.getElementById('timeline_body');
let offsetX = Timeline.vue.scroll_left + (body.clientWidth - Timeline.vue.head_width) / 2;
if (mode == 'reset') {
let original_size = Timeline.vue._data.size
Timeline.vue._data.size = 200;
body.scrollLeft += (Timeline.vue._data.size - original_size) * (offsetX / original_size)
} else {
let zoom = mode == 'in' ? 1.2 : 0.8;
let original_size = Timeline.vue._data.size
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)
}
} else {
2019-07-18 00:02:07 +08:00
switch (mode) {
case 'in': Preview.selected.controls.dollyIn(1.16); break;
case 'out': Preview.selected.controls.dollyOut(1.16); break;
2019-07-18 00:02:07 +08:00
}
}
2019-07-18 00:02:07 +08:00
}
//UI Edit
function setProgressBar(id, val, time) {
if (!id || id === 'main') {
Prop.progress = val
} else {
$('#'+id+' > .progress_bar_inner').animate({width: val*488}, time-1)
}
if (isApp) {
currentwindow.setProgressBar(val)
}
}
//Tooltip
function showShiftTooltip() {
$(':hover').find('.tooltip_shift').css('display', 'inline')
}
$(document).keyup(function(event) {
if (event.which === 16) {
$('.tooltip_shift').hide()
}
})
2022-02-23 01:12:27 +08:00
/**
*
* @param {string} tag Tag name
* @param {object} [attributes] Attributes
* @param {string|HTMLElement|string[]|HTMLElement[]} [content] Content
* @returns {HTMLElement} Element
*/
Interface.createElement = (tag, attributes = {}, content) => {
let el = document.createElement(tag);
for (let key in attributes) {
if (attributes[key] !== undefined) {
el.setAttribute(key, attributes[key]);
}
}
if (typeof content == 'string') el.textContent = content;
if (content instanceof Array) {
content.forEach(node => el.append(node));
}
if (content instanceof HTMLElement) el.append(content);
return el;
}
2019-07-18 00:02:07 +08:00
// Custom Elements
Interface.CustomElements.ResizeLine = ResizeLine;
Interface.CustomElements.SelectInput = function(id, data) {
function getNameFor(key) {
let val = data.options[key];
if (val) {
return tl(val.name || val);
} else {
return '';
}
}
let value = data.value || data.default || Object.keys(data.options)[0];
let select = Interface.createElement('bb-select', {id, class: 'half', value: value}, getNameFor(value));
function setKey(key) {
value = key;
select.setAttribute('value', key);
select.textContent = getNameFor(key);
if (typeof data.onChange == 'function') {
data.onChange(value);
}
}
select.addEventListener('click', function(event) {
if (Menu.closed_in_this_click == id) return this;
let items = [];
for (let key in data.options) {
let val = data.options[key];
if (val) {
items.push({
name: getNameFor(key),
icon: val.icon || ((value == key) ? 'far.fa-dot-circle' : 'far.fa-circle'),
condition: val.condition,
click: (e) => {
setKey(key);
}
})
}
}
let menu = new Menu(id, items, {searchable: items.length > 16});
menu.node.style['min-width'] = select.clientWidth+'px';
menu.open(select);
})
this.node = select;
this.set = setKey;
}
2022-11-27 21:17:17 +08:00
function openTouchKeyboardModifierMenu(node) {
if (Menu.closed_in_this_click == 'mobile_keyboard') return;
let modifiers = ['ctrl', 'shift', 'alt'];
let menu = new Menu('mobile_keyboard', [
...modifiers.map(key => {
let name = tl(`keys.${key}`);
if (Interface.status_bar.vue.modifier_keys[key].length) {
name += ' (' + tl(Interface.status_bar.vue.modifier_keys[key].last()) + ')';
}
return {
name,
icon: Pressing.overrides[key] ? 'check_box' : 'check_box_outline_blank',
click() {
Pressing.overrides[key] = !Pressing.overrides[key]
}
}
}),
'_',
{icon: 'clear_all', name: 'menu.mobile_keyboard.disable_all', condition: () => {
let {length} = [Pressing.overrides.ctrl, Pressing.overrides.shift, Pressing.overrides.alt].filter(key => key);
return length;
}, click() {
Pressing.overrides.ctrl = false; Pressing.overrides.shift = false; Pressing.overrides.alt = false;
}},
])
menu.open(node);
}
2019-07-18 00:02:07 +08:00
onVueSetup(function() {
2021-01-24 21:39:53 +08:00
Interface.status_bar.vue = new Vue({
2020-12-22 20:32:49 +08:00
el: '#status_bar',
2019-07-18 00:02:07 +08:00
data: {
Prop,
isMobile: Blockbench.isMobile,
streamer_mode: settings.streamer_mode.value,
selection_info: '',
Format: null,
2021-11-30 06:50:51 +08:00
show_modifier_keys: settings.status_bar_modifier_keys.value,
2022-08-04 03:14:59 +08:00
warnings: Validator.warnings,
errors: Validator.errors,
modifier_keys: {
ctrl: [],
shift: [],
alt: []
2022-11-27 21:17:17 +08:00
},
modifiers: Blockbench.isTouch && !Blockbench.isMobile && Pressing.overrides,
keyboard_menu_in_status_bar: Blockbench.isTouch && !Blockbench.isMobile
2021-01-24 21:39:53 +08:00
},
methods: {
showContextMenu(event) {
Interface.status_bar.menu.show(event);
},
toggleStreamerMode() {
ActionControl.select(`setting: ${tl('settings.streamer_mode')}`);
},
updateSelectionInfo() {
let selection_mode = BarItems.selection_mode.value;
if (Modes.edit && Mesh.selected.length && selection_mode !== 'object') {
if (selection_mode == 'face') {
let total = 0, selected = 0;
Mesh.selected.forEach(mesh => {
total += Object.keys(mesh.faces).length;
selected += mesh.getSelectedFaces().length;
});
this.selection_info = tl('status_bar.selection.faces', `${selected} / ${total}`);
}
if (selection_mode == 'edge') {
let total = 0, selected = 0;
Mesh.selected.forEach(mesh => {
let processed_lines = [];
mesh.forAllFaces(face => {
let vertices = face.getSortedVertices();
vertices.forEach((vkey, i) => {
let vkey2 = vertices[i+1] || vertices[0];
if (!processed_lines.find(processed => processed.includes(vkey) && processed.includes(vkey2))) {
processed_lines.push([vkey, vkey2]);
total += 1;
}
})
})
selected += mesh.getSelectedEdges().length;
})
this.selection_info = tl('status_bar.selection.edges', `${selected} / ${total}`);
}
if (selection_mode == 'vertex') {
let total = 0, selected = 0;
Mesh.selected.forEach(mesh => total += Object.keys(mesh.vertices).length);
Mesh.selected.forEach(mesh => selected += mesh.getSelectedVertices().length);
this.selection_info = tl('status_bar.selection.vertices', `${selected} / ${total}`);
}
} else {
this.selection_info = '';
}
},
2021-11-30 06:50:51 +08:00
clickModifiers() {
ActionControl.select(`setting: ${tl('settings.status_bar_modifier_keys')}`);
},
2022-08-04 03:14:59 +08:00
openValidator() {
Validator.openDialog();
},
2022-11-27 21:17:17 +08:00
openKeyboardMenu() {
openTouchKeyboardModifierMenu(this.$refs.mobile_keyboard_menu);
},
toggleSidebar: Interface.toggleSidebar,
2021-11-30 06:50:51 +08:00
getIconNode: Blockbench.getIconNode,
tl
2021-01-24 21:39:53 +08:00
},
template: `
<div id="status_bar" @contextmenu="showContextMenu($event)">
<div class="sidebar_toggle_button" v-if="!isMobile" @click="toggleSidebar('left')" title="${tl('status_bar.toggle_sidebar')}">
2021-01-24 21:39:53 +08:00
<i class="material-icons">{{Prop.show_left_bar ? 'chevron_left' : 'chevron_right'}}</i>
</div>
<div class="f_left" v-if="streamer_mode"
2021-01-24 21:39:53 +08:00
style="background-color: var(--color-stream); color: var(--color-light);"
@click="toggleStreamerMode()"
title="${tl('interface.streamer_mode_on')}"
2021-01-24 21:39:53 +08:00
>
<i class="material-icons">live_tv</i>
</div>
<div v-if="Format" v-html="getIconNode(Format.icon).outerHTML" v-bind:title="Format.name"></div>
<div v-if="Prop.recording" v-html="getIconNode('fiber_manual_record').outerHTML" style="color: var(--color-close)" title="${tl('status_bar.recording')}"></div>
2021-01-24 21:39:53 +08:00
<div id="status_name">
{{ Prop.file_name }}
</div>
<div id="status_message" class="hidden"></div>
2021-11-30 06:50:51 +08:00
<template v-if="show_modifier_keys && !isMobile">
2021-11-30 06:50:51 +08:00
<div class="status_bar_modifier_key" v-if="modifier_keys.ctrl.length" @click="clickModifiers()">
<kbd>${tl(Blockbench.platform == 'darwin' ? 'keys.meta' : 'keys.ctrl')}</kbd>
2021-11-30 06:50:51 +08:00
<span>{{ tl(modifier_keys.ctrl.last()) }}</span>
</div>
<div class="status_bar_modifier_key" v-if="modifier_keys.shift.length" @click="clickModifiers()">
<kbd>${tl('keys.shift')}</kbd>
<span>{{ tl(modifier_keys.shift.last()) }}</span>
</div>
<div class="status_bar_modifier_key" v-if="modifier_keys.alt.length" @click="clickModifiers()">
<kbd>${tl('keys.alt')}</kbd>
<span>{{ tl(modifier_keys.alt.last()) }}</span>
</div>
</template>
<div class="status_selection_info">{{ selection_info }}</div>
2022-08-04 03:14:59 +08:00
<div class="f_right" id="validator_status" v-if="warnings.length || errors.length" @click="openValidator()">
<span v-if="warnings.length" style="color: var(--color-warning)">{{ warnings.length }}<i class="material-icons">warning</i></span>
<span v-if="errors.length" style="color: var(--color-error)">{{ errors.length }}<i class="material-icons">error</i></span>
</div>
2022-11-27 21:17:17 +08:00
<div v-if="keyboard_menu_in_status_bar" id="mobile_keyboard_menu" @click="openKeyboardMenu()" ref="mobile_keyboard_menu" :class="{enabled: modifiers.ctrl || modifiers.shift || modifiers.alt}">
<i class="material-icons">keyboard</i>
</div>
2021-01-24 21:39:53 +08:00
<div class="f_right">
{{ Prop.fps }} FPS
</div>
<div class="sidebar_toggle_button" v-if="!isMobile" @click="toggleSidebar('right')" title="${tl('status_bar.toggle_sidebar')}">
2021-01-24 21:39:53 +08:00
<i class="material-icons">{{Prop.show_right_bar ? 'chevron_right' : 'chevron_left'}}</i>
</div>
<div id="status_progress" v-if="Prop.progress" v-bind:style="{width: Prop.progress*100+'%'}"></div>
</div>
`
2019-07-18 00:02:07 +08:00
})
2021-11-30 06:50:51 +08:00
Interface.addSuggestedModifierKey = (key, text) => {
Interface.status_bar.vue.modifier_keys[key].safePush(text);
};
Interface.removeSuggestedModifierKey = (key, text) => {
Interface.status_bar.vue.modifier_keys[key].remove(text);
};
2019-07-18 00:02:07 +08:00
})