2020-07-16 15:32:59 +08:00
|
|
|
class Panel {
|
2022-02-23 01:12:27 +08:00
|
|
|
constructor(id, data) {
|
|
|
|
if (!data) data = id;
|
2020-07-16 15:32:59 +08:00
|
|
|
let scope = this;
|
|
|
|
this.type = 'panel';
|
2022-02-23 01:12:27 +08:00
|
|
|
this.id = typeof id == 'string' ? id : data.id || 'new_panel';
|
|
|
|
this.name = tl(data.name ? data.name : `panel.${this.id}`);
|
2020-07-16 15:32:59 +08:00
|
|
|
this.icon = data.icon;
|
|
|
|
this.menu = data.menu;
|
2022-02-23 01:12:27 +08:00
|
|
|
this.condition = data.condition;
|
2022-03-07 21:37:19 +08:00
|
|
|
this.display_condition = data.display_condition;
|
2022-02-23 01:12:27 +08:00
|
|
|
this.previous_slot = 'left_bar';
|
|
|
|
|
2020-09-14 00:31:46 +08:00
|
|
|
this.growable = data.growable;
|
2020-07-16 15:32:59 +08:00
|
|
|
this.selection_only = data.selection_only == true;
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2020-07-16 15:32:59 +08:00
|
|
|
this.onResize = data.onResize;
|
2021-09-15 23:50:46 +08:00
|
|
|
this.onFold = data.onFold;
|
2022-03-13 05:57:54 +08:00
|
|
|
this.events = {};
|
2022-02-23 01:12:27 +08:00
|
|
|
this.toolbars = data.toolbars || {};
|
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
if (!Interface.data.panels[this.id]) Interface.data.panels[this.id] = {};
|
2022-02-23 01:12:27 +08:00
|
|
|
this.position_data = Interface.data.panels[this.id];
|
2022-03-04 04:13:30 +08:00
|
|
|
let defaultp = data.default_position || 0;
|
2022-03-09 06:09:31 +08:00
|
|
|
if (defaultp && defaultp.slot) this.previous_slot = defaultp.slot;
|
2022-03-04 04:13:30 +08:00
|
|
|
if (!this.position_data.slot) this.position_data.slot = defaultp.slot || (data.default_side ? (data.default_side+'_bar') : 'left_bar');
|
|
|
|
if (!this.position_data.float_position) this.position_data.float_position = defaultp.float_position || [0, 0];
|
|
|
|
if (!this.position_data.float_size) this.position_data.float_size = defaultp.float_size || [300, 300];
|
|
|
|
if (!this.position_data.height) this.position_data.height = defaultp.height || 300;
|
2022-03-13 21:21:26 +08:00
|
|
|
if (this.position_data.folded == undefined) this.position_data.folded = defaultp.folded || false;
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2022-03-07 21:37:19 +08:00
|
|
|
this.handle = Interface.createElement('h3', {class: 'panel_handle'}, Interface.createElement('label', {}, Interface.createElement('span', {}, this.name)));
|
2022-02-23 01:12:27 +08:00
|
|
|
this.node = Interface.createElement('div', {class: 'panel', id: `panel_${this.id}`}, this.handle);
|
|
|
|
|
|
|
|
if (this.selection_only) this.node.classList.add('selection_only');
|
|
|
|
if (this.growable) this.node.classList.add('grow');
|
|
|
|
|
|
|
|
// Toolbars
|
|
|
|
for (let key in this.toolbars) {
|
|
|
|
let toolbar = this.toolbars[key];
|
|
|
|
if (toolbar instanceof Toolbar) {
|
|
|
|
if (toolbar.label) {
|
|
|
|
let label = Interface.createElement('p', {class: 'panel_toolbar_label'}, tl(toolbar.name));
|
|
|
|
this.node.append(label);
|
|
|
|
}
|
|
|
|
this.node.append(toolbar.node);
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2020-10-05 05:44:06 +08:00
|
|
|
if (data.component) {
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
let component_mount = Interface.createElement('div');
|
2022-02-23 01:12:27 +08:00
|
|
|
this.node.append(component_mount);
|
2022-03-04 04:13:30 +08:00
|
|
|
let onmounted = data.component.mounted;
|
2022-02-23 01:12:27 +08:00
|
|
|
data.component.mounted = function() {
|
|
|
|
Vue.nextTick(() => {
|
2020-09-26 18:04:40 +08:00
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
let toolbar_wrappers = this.$el.querySelectorAll('.toolbar_wrapper');
|
|
|
|
toolbar_wrappers.forEach(wrapper => {
|
|
|
|
let id = wrapper.attributes.toolbar && wrapper.attributes.toolbar.value;
|
|
|
|
let toolbar = scope.toolbars[id];
|
|
|
|
if (toolbar) {
|
|
|
|
wrapper.append(toolbar.node);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
if (typeof onmounted == 'function') {
|
|
|
|
onmounted.call(this);
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
2022-03-07 21:37:19 +08:00
|
|
|
//updateInterfacePanels()
|
2022-02-23 01:12:27 +08:00
|
|
|
})
|
2020-09-26 18:04:40 +08:00
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
this.vue = this.inside_vue = new Vue(data.component).$mount(component_mount);
|
|
|
|
scope.vue.$el.classList.add('panel_vue_wrapper');
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!Blockbench.isMobile) {
|
2022-03-09 06:09:31 +08:00
|
|
|
if (data.expand_button) {
|
|
|
|
let expand_button = Interface.createElement('div', {class: 'tool panel_control panel_expanding_button'}, Blockbench.getIconNode('fullscreen'))
|
|
|
|
this.handle.append(expand_button);
|
|
|
|
expand_button.addEventListener('click', (e) => {
|
|
|
|
if (this.slot == 'float') {
|
|
|
|
this.moveTo(this.previous_slot);
|
|
|
|
} else {
|
|
|
|
this.moveTo('float');
|
|
|
|
this.moveToFront();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
|
|
|
|
let snap_button = Interface.createElement('div', {class: 'tool panel_control'}, Blockbench.getIconNode('drag_handle'))
|
|
|
|
this.handle.append(snap_button);
|
|
|
|
snap_button.addEventListener('click', (e) => {
|
|
|
|
new Menu([
|
|
|
|
{
|
2022-03-04 04:13:30 +08:00
|
|
|
name: 'Left Sidebar',
|
|
|
|
icon: 'align_horizontal_left',
|
|
|
|
click: () => this.moveTo('left_bar')
|
2022-02-23 01:12:27 +08:00
|
|
|
},
|
|
|
|
{
|
2022-03-04 04:13:30 +08:00
|
|
|
name: 'Right Sidebar',
|
|
|
|
icon: 'align_horizontal_right',
|
|
|
|
click: () => this.moveTo('right_bar')
|
2022-02-23 01:12:27 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Top',
|
2022-03-04 04:13:30 +08:00
|
|
|
icon: 'align_vertical_top',
|
|
|
|
click: () => this.moveTo('top')
|
2022-02-23 01:12:27 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Bottom',
|
2022-03-04 04:13:30 +08:00
|
|
|
icon: 'align_vertical_bottom',
|
|
|
|
click: () => this.moveTo('bottom')
|
2022-02-23 01:12:27 +08:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'Float',
|
2022-03-04 04:13:30 +08:00
|
|
|
icon: 'web_asset',
|
|
|
|
click: () => this.moveTo('float')
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
|
|
|
]).show(snap_button);
|
|
|
|
})
|
|
|
|
|
|
|
|
let fold_button = Interface.createElement('div', {class: 'tool panel_control panel_folding_button'}, Blockbench.getIconNode('expand_more'))
|
2021-03-09 00:50:59 +08:00
|
|
|
this.handle.append(fold_button);
|
|
|
|
fold_button.addEventListener('click', (e) => {
|
|
|
|
this.fold();
|
|
|
|
})
|
|
|
|
|
2022-03-05 06:34:36 +08:00
|
|
|
this.handle.firstElementChild.addEventListener('dblclick', e => {
|
|
|
|
this.fold();
|
|
|
|
})
|
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
|
|
|
|
|
|
|
|
addEventListeners(this.handle.firstElementChild, 'mousedown touchstart', e1 => {
|
|
|
|
convertTouchEvent(e1);
|
|
|
|
let started = false;
|
|
|
|
let position_before = this.slot == 'float'
|
|
|
|
? this.position_data.float_position.slice()
|
|
|
|
: [e1.clientX - e1.offsetX, e1.clientY - e1.offsetY - 55];
|
2022-03-11 01:12:59 +08:00
|
|
|
let original_show_left_bar = Prop.show_left_bar;
|
|
|
|
let original_show_right_bar = Prop.show_right_bar;
|
|
|
|
|
|
|
|
let target_slot, target_panel, target_before;
|
|
|
|
function updateTargetHighlight() {
|
|
|
|
$(`.panel[order]`).attr('order', null);
|
|
|
|
if (target_panel) target_panel.node.setAttribute('order', target_before ? -1 : 1);
|
|
|
|
|
|
|
|
if (target_slot) {
|
|
|
|
Interface.center_screen.setAttribute('snapside', target_slot);
|
|
|
|
} else {
|
|
|
|
Interface.center_screen.removeAttribute('snapside');
|
|
|
|
}
|
|
|
|
if ((target_slot == 'right_bar' && Interface.right_bar_width) || (target_slot == 'left_bar' && Interface.left_bar_width)) {
|
|
|
|
Interface.center_screen.removeAttribute('snapside');
|
|
|
|
}
|
2022-03-11 01:34:00 +08:00
|
|
|
Interface.left_bar.classList.toggle('drop_target', target_slot == 'left_bar');
|
|
|
|
Interface.right_bar.classList.toggle('drop_target', target_slot == 'right_bar');
|
|
|
|
|
2022-03-11 01:12:59 +08:00
|
|
|
if (target_slot == 'left_bar' && !Prop.show_left_bar) Interface.toggleSidebar('left');
|
|
|
|
if (target_slot == 'right_bar' && !Prop.show_right_bar) Interface.toggleSidebar('right');
|
|
|
|
if (target_slot != 'left_bar' && Prop.show_left_bar && !original_show_left_bar) Interface.toggleSidebar('left');
|
|
|
|
if (target_slot != 'right_bar' && Prop.show_right_bar && !original_show_right_bar) Interface.toggleSidebar('right');
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
|
|
|
|
let drag = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
2022-03-18 06:00:30 +08:00
|
|
|
if (!started && (Math.pow(e2.clientX - e1.clientX, 2) + Math.pow(e2.clientY - e1.clientY, 2)) > 15) {
|
2022-02-23 01:12:27 +08:00
|
|
|
started = true;
|
2022-03-05 06:34:36 +08:00
|
|
|
if (this.slot !== 'float') {
|
|
|
|
this.moveTo('float');
|
|
|
|
this.moveToFront();
|
|
|
|
}
|
2022-03-10 22:17:29 +08:00
|
|
|
this.node.classList.add('dragging');
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
|
|
|
if (!started) return;
|
2022-03-05 06:34:36 +08:00
|
|
|
|
|
|
|
this.position_data.float_position[0] = position_before[0] + e2.clientX - e1.clientX;
|
|
|
|
this.position_data.float_position[1] = position_before[1] + e2.clientY - e1.clientY;
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2022-03-11 01:12:59 +08:00
|
|
|
let threshold = 40;
|
2022-03-11 04:24:33 +08:00
|
|
|
let threshold_y = 64;
|
2022-03-11 01:12:59 +08:00
|
|
|
target_slot = null; target_panel = null; target_before = false;
|
|
|
|
|
2022-03-11 19:34:53 +08:00
|
|
|
if (e2.ctrlOrCmd) {
|
|
|
|
} else if (e2.clientX < Math.max(Interface.left_bar_width, threshold)) {
|
2022-03-11 01:12:59 +08:00
|
|
|
|
|
|
|
let y = Interface.work_screen.offsetTop;
|
|
|
|
target_slot = 'left_bar';
|
|
|
|
for (let child of Interface.left_bar.childNodes) {
|
|
|
|
if (!child.clientHeight) continue;
|
|
|
|
target_panel = Panels[child.id.replace(/^panel_/, '')];
|
|
|
|
if (e2.clientY < (y + child.clientHeight / 2)) {
|
|
|
|
target_before = true;
|
|
|
|
break;
|
2022-03-05 06:34:36 +08:00
|
|
|
}
|
2022-03-11 01:12:59 +08:00
|
|
|
y += child.clientHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (e2.clientX > document.body.clientWidth - Math.max(Interface.right_bar_width, threshold)) {
|
|
|
|
|
|
|
|
let y = Interface.work_screen.offsetTop + 30;
|
|
|
|
target_slot = 'right_bar';
|
|
|
|
for (let child of Interface.right_bar.childNodes) {
|
|
|
|
if (!child.clientHeight) continue;
|
|
|
|
target_panel = Panels[child.id.replace(/^panel_/, '')];
|
|
|
|
if (e2.clientY < (y + child.clientHeight / 2)) {
|
|
|
|
target_before = true;
|
|
|
|
break;
|
2022-03-05 06:34:36 +08:00
|
|
|
}
|
2022-03-11 01:12:59 +08:00
|
|
|
y += child.clientHeight;
|
|
|
|
}
|
2022-03-05 06:34:36 +08:00
|
|
|
|
2022-03-07 21:37:19 +08:00
|
|
|
} else if (
|
2022-03-11 04:24:33 +08:00
|
|
|
e2.clientY < (Interface.work_screen.offsetTop + 30 + threshold_y) &&
|
2022-03-11 01:12:59 +08:00
|
|
|
e2.clientX > Interface.left_bar_width && e2.clientX < (Interface.work_screen.clientWidth - Interface.right_bar_width)
|
2022-03-07 21:37:19 +08:00
|
|
|
) {
|
2022-03-11 01:12:59 +08:00
|
|
|
target_slot = 'top'
|
2022-03-05 06:34:36 +08:00
|
|
|
|
2022-03-07 21:37:19 +08:00
|
|
|
} else if (
|
2022-03-11 04:24:33 +08:00
|
|
|
e2.clientY > Interface.work_screen.offsetTop + Interface.work_screen.clientHeight - Interface.status_bar.vue.$el.clientHeight - threshold_y &&
|
2022-03-11 01:12:59 +08:00
|
|
|
e2.clientX > Interface.left_bar_width && e2.clientX < (Interface.work_screen.clientWidth - Interface.right_bar_width)
|
2022-03-07 21:37:19 +08:00
|
|
|
) {
|
2022-03-11 01:12:59 +08:00
|
|
|
target_slot = 'bottom'
|
2022-03-05 06:34:36 +08:00
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
}
|
2022-03-11 01:12:59 +08:00
|
|
|
updateTargetHighlight();
|
2022-03-05 06:34:36 +08:00
|
|
|
this.update(true);
|
2022-03-13 05:57:54 +08:00
|
|
|
this.dispatchEvent('drag', {event: e2, target_before, target_panel, target_slot});
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
|
|
|
let stop = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
2022-03-10 22:17:29 +08:00
|
|
|
this.node.classList.remove('dragging');
|
2022-03-11 01:12:59 +08:00
|
|
|
Interface.center_screen.removeAttribute('snapside');
|
|
|
|
$(`.panel[order]`).attr('order', null);
|
2022-03-11 01:34:00 +08:00
|
|
|
Interface.left_bar.classList.remove('drop_target');
|
|
|
|
Interface.right_bar.classList.remove('drop_target');
|
2022-03-11 01:12:59 +08:00
|
|
|
|
|
|
|
if (target_slot) {
|
|
|
|
this.moveTo(target_slot, target_panel, target_before)
|
|
|
|
}
|
2022-03-10 22:17:29 +08:00
|
|
|
|
2022-03-05 06:34:36 +08:00
|
|
|
if (this.slot != 'float') {
|
|
|
|
this.position_data.float_position[0] = position_before[0];
|
|
|
|
this.position_data.float_position[1] = position_before[1];
|
|
|
|
}
|
|
|
|
this.update();
|
2022-02-23 01:12:27 +08:00
|
|
|
updateInterface()
|
|
|
|
|
|
|
|
removeEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
removeEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
}
|
|
|
|
addEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
addEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
|
|
|
|
})
|
2022-03-07 21:37:19 +08:00
|
|
|
} else {
|
2022-02-23 01:12:27 +08:00
|
|
|
|
|
|
|
let close_button = Interface.createElement('div', {class: 'tool panel_control'}, Blockbench.getIconNode('clear'))
|
|
|
|
this.handle.append(close_button);
|
|
|
|
close_button.addEventListener('click', (e) => {
|
|
|
|
Interface.PanelSelectorVue.select(null);
|
2020-07-16 15:32:59 +08:00
|
|
|
})
|
2022-03-07 21:37:19 +08:00
|
|
|
|
|
|
|
|
|
|
|
addEventListeners(this.handle.firstElementChild, 'mousedown touchstart', e1 => {
|
|
|
|
convertTouchEvent(e1);
|
|
|
|
let started = false;
|
|
|
|
let height_before = this.position_data.height;
|
|
|
|
let max = Blockbench.isLandscape ? window.innerWidth - 50 : Interface.work_screen.clientHeight;
|
|
|
|
|
|
|
|
let drag = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
|
|
|
let diff = Blockbench.isLandscape ? e1.clientX - e2.clientX : e1.clientY - e2.clientY;
|
|
|
|
if (!started && Math.abs(diff) > 4) {
|
|
|
|
started = true;
|
|
|
|
}
|
|
|
|
if (!started) return;
|
|
|
|
|
2022-03-16 23:20:27 +08:00
|
|
|
let sign = settings.mobile_panel_side.value == 'left' ? -1 : 1;
|
|
|
|
this.position_data.height = Math.clamp(height_before + diff * sign, 140, max);
|
2022-03-07 21:37:19 +08:00
|
|
|
|
|
|
|
this.update(true);
|
|
|
|
resizeWindow();
|
|
|
|
|
|
|
|
}
|
|
|
|
let stop = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
|
|
|
|
|
|
|
this.update();
|
|
|
|
|
|
|
|
removeEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
removeEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
}
|
|
|
|
addEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
addEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
|
|
|
|
})
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
|
|
|
this.node.addEventListener('mousedown', event => {
|
2022-03-04 04:13:30 +08:00
|
|
|
setActivePanel(this.id);
|
|
|
|
this.moveToFront();
|
2022-02-23 01:12:27 +08:00
|
|
|
})
|
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
|
|
|
|
// Add to slot
|
2022-03-15 01:20:54 +08:00
|
|
|
if (Blockbench.isMobile) {
|
|
|
|
this.moveTo('left_bar');
|
|
|
|
} else {
|
|
|
|
let reference_panel = Panels[data.insert_before || data.insert_after];
|
|
|
|
this.moveTo(this.position_data.slot, reference_panel, reference_panel && !data.insert_after);
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-03-13 21:21:26 +08:00
|
|
|
if (this.folded) this.fold(true);
|
|
|
|
|
2022-03-11 01:34:00 +08:00
|
|
|
Panels[this.id] = this;
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2021-09-15 23:50:46 +08:00
|
|
|
isVisible() {
|
|
|
|
return !this.folded && this.node.parentElement && this.node.parentElement.style.display !== 'none';
|
|
|
|
}
|
2022-03-16 23:20:27 +08:00
|
|
|
isInSidebar() {
|
|
|
|
return this.slot === 'left_bar' || this.slot === 'right_bar';
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
get slot() {
|
|
|
|
return this.position_data.slot;
|
|
|
|
}
|
2022-03-13 21:21:26 +08:00
|
|
|
get folded() {
|
|
|
|
return this.position_data.folded;
|
|
|
|
}
|
|
|
|
set folded(state) {
|
|
|
|
this.position_data.folded = !!state;
|
|
|
|
}
|
2021-03-03 05:24:23 +08:00
|
|
|
fold(state = !this.folded) {
|
|
|
|
this.folded = !!state;
|
2022-02-23 01:12:27 +08:00
|
|
|
let new_icon = Blockbench.getIconNode(state ? 'expand_less' : 'expand_more');
|
|
|
|
$(this.handle).find('> .panel_folding_button > .icon').replaceWith(new_icon);
|
2021-03-03 05:24:23 +08:00
|
|
|
this.node.classList.toggle('folded', state);
|
2021-09-15 23:50:46 +08:00
|
|
|
if (this.onFold) {
|
|
|
|
this.onFold();
|
|
|
|
}
|
2022-03-07 21:37:19 +08:00
|
|
|
if (this.slot == 'top' || this.slot == 'bottom') {
|
|
|
|
resizeWindow();
|
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
this.update();
|
2022-03-13 05:57:54 +08:00
|
|
|
this.dispatchEvent('fold', {});
|
2022-03-04 04:13:30 +08:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
setupFloatHandles() {
|
|
|
|
let sides = [
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_side resize_top'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_side resize_bottom'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_side resize_left'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_side resize_right'}),
|
|
|
|
];
|
|
|
|
let corners = [
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_corner resize_top_left'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_corner resize_top_right'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_corner resize_bottom_left'}),
|
|
|
|
Interface.createElement('div', {class: 'panel_resize_corner resize_bottom_right'}),
|
|
|
|
];
|
|
|
|
let resize = (e1, direction_x, direction_y) => {
|
|
|
|
let position_before = this.position_data.float_position.slice();
|
2022-03-11 19:34:53 +08:00
|
|
|
let size_before = [this.width, this.height];
|
2022-03-04 04:13:30 +08:00
|
|
|
let started = false;
|
|
|
|
|
|
|
|
let drag = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
2022-03-11 19:34:53 +08:00
|
|
|
if (!started && (Math.pow(e2.clientX - e1.clientX, 2) + Math.pow(e2.clientY - e1.clientY, 2)) > 12) {
|
2022-03-04 04:13:30 +08:00
|
|
|
started = true;
|
|
|
|
}
|
|
|
|
if (!started) return;
|
|
|
|
|
|
|
|
this.position_data.float_size[0] = size_before[0] + (e2.clientX - e1.clientX) * direction_x;
|
|
|
|
this.position_data.float_size[1] = size_before[1] + (e2.clientY - e1.clientY) * direction_y;
|
|
|
|
|
|
|
|
if (direction_x == -1) this.position_data.float_position[0] = position_before[0] - this.position_data.float_size[0] + size_before[0];
|
|
|
|
if (direction_y == -1) this.position_data.float_position[1] = position_before[1] - this.position_data.float_size[1] + size_before[1];
|
|
|
|
|
|
|
|
this.update();
|
|
|
|
}
|
|
|
|
let stop = e2 => {
|
|
|
|
convertTouchEvent(e2);
|
|
|
|
|
|
|
|
removeEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
removeEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
}
|
|
|
|
addEventListeners(document, 'mousemove touchmove', drag);
|
|
|
|
addEventListeners(document, 'mouseup touchend', stop);
|
|
|
|
}
|
|
|
|
addEventListeners(sides[0], 'mousedown touchstart', (event) => resize(event, 0, -1));
|
|
|
|
addEventListeners(sides[1], 'mousedown touchstart', (event) => resize(event, 0, 1));
|
|
|
|
addEventListeners(sides[2], 'mousedown touchstart', (event) => resize(event, -1, 0));
|
|
|
|
addEventListeners(sides[3], 'mousedown touchstart', (event) => resize(event, 1, 0));
|
|
|
|
addEventListeners(corners[0], 'mousedown touchstart', (event) => resize(event, -1, -1));
|
|
|
|
addEventListeners(corners[1], 'mousedown touchstart', (event) => resize(event, 1, -1));
|
|
|
|
addEventListeners(corners[2], 'mousedown touchstart', (event) => resize(event, -1, 1));
|
|
|
|
addEventListeners(corners[3], 'mousedown touchstart', (event) => resize(event, 1, 1));
|
|
|
|
|
|
|
|
let handles = Interface.createElement('div', {class: 'panel_resize_handle_wrapper'}, [...sides, ...corners]);
|
|
|
|
this.node.append(handles);
|
|
|
|
this.resize_handles = handles;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
moveToFront() {
|
|
|
|
if (this.slot == 'float' && Panel.floating_panel_z_order[0] !== this.id) {
|
|
|
|
Panel.floating_panel_z_order.remove(this.id);
|
|
|
|
Panel.floating_panel_z_order.splice(0, 0, this.id);
|
|
|
|
let zindex = 18;
|
|
|
|
Panel.floating_panel_z_order.forEach(id => {
|
|
|
|
let panel = Panels[id];
|
|
|
|
panel.node.style.zIndex = zindex;
|
2022-03-16 23:20:27 +08:00
|
|
|
panel.dispatchEvent('change_zindex', {zindex});
|
2022-03-04 04:13:30 +08:00
|
|
|
zindex = Math.clamp(zindex-1, 14, 19);
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return this;
|
2021-03-03 05:24:23 +08:00
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
moveTo(slot, ref_panel, before = false) {
|
|
|
|
let position_data = this.position_data;
|
|
|
|
if (slot == undefined) {
|
|
|
|
slot = ref_panel.position_data.slot;
|
|
|
|
}
|
2022-03-09 06:09:31 +08:00
|
|
|
if (slot !== this.slot) {
|
|
|
|
this.previous_slot = this.slot;
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
|
2022-03-13 05:57:54 +08:00
|
|
|
this.dispatchEvent('move_to', {slot, ref_panel, before, previous_slot: this.previous_slot});
|
|
|
|
|
|
|
|
this.node.classList.remove('floating');
|
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
if (slot == 'left_bar' || slot == 'right_bar') {
|
2022-03-12 22:09:09 +08:00
|
|
|
let change_panel_order = !!ref_panel;
|
2022-03-04 04:13:30 +08:00
|
|
|
if (!ref_panel && Interface.data[slot].includes(this.id)) {
|
2022-03-11 01:34:00 +08:00
|
|
|
let panels = Interface.data[slot].filter(id => Panels[id] && Panels[id].slot == slot || id == this.id);
|
|
|
|
let index = panels.indexOf(this.id);
|
2022-03-04 04:13:30 +08:00
|
|
|
if (index == 0) {
|
2022-03-11 01:34:00 +08:00
|
|
|
ref_panel = Panels[panels[1]];
|
2022-03-04 04:13:30 +08:00
|
|
|
before = true;
|
|
|
|
} else {
|
2022-03-11 01:34:00 +08:00
|
|
|
ref_panel = Panels[panels[index-1]];
|
2022-03-04 04:13:30 +08:00
|
|
|
before = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
if (ref_panel instanceof Panel && ref_panel.slot == slot) {
|
|
|
|
if (before) {
|
|
|
|
$(ref_panel.node).before(this.node);
|
|
|
|
} else {
|
|
|
|
$(ref_panel.node).after(this.node);
|
|
|
|
}
|
2022-03-12 22:09:09 +08:00
|
|
|
if (change_panel_order) {
|
|
|
|
Interface.data[slot].remove(this.id);
|
|
|
|
Interface.data[slot].splice(Interface.data[slot].indexOf(ref_panel.id) + (before ? 0 : 1), 0, this.id);
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
} else {
|
2022-03-04 04:13:30 +08:00
|
|
|
document.getElementById(slot).append(this.node);
|
2022-03-09 06:09:31 +08:00
|
|
|
Interface.data[slot].safePush(this.id);
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
|
|
|
|
} else if (slot == 'top') {
|
|
|
|
let top_panel = Interface.getTopPanel();
|
2022-03-09 02:23:30 +08:00
|
|
|
if (top_panel && top_panel !== this && !Condition.mutuallyExclusive(this.condition, top_panel.condition)) {
|
|
|
|
top_panel.moveTo(top_panel.previous_slot);
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
document.getElementById('top_slot').append(this.node);
|
|
|
|
|
|
|
|
} else if (slot == 'bottom') {
|
|
|
|
let bottom_panel = Interface.getBottomPanel();
|
2022-03-09 02:23:30 +08:00
|
|
|
if (bottom_panel && bottom_panel !== this && !Condition.mutuallyExclusive(this.condition, bottom_panel.condition)) {
|
|
|
|
bottom_panel.moveTo(bottom_panel.previous_slot);
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
document.getElementById('bottom_slot').append(this.node);
|
|
|
|
|
2022-03-10 22:17:29 +08:00
|
|
|
} else if (slot == 'float' && !Blockbench.isMobile) {
|
2022-03-05 06:34:36 +08:00
|
|
|
Interface.work_screen.append(this.node);
|
2022-02-23 01:12:27 +08:00
|
|
|
this.node.classList.add('floating');
|
2022-03-16 23:20:27 +08:00
|
|
|
this.dispatchEvent('change_zindex', {zindex: 14});
|
2022-03-04 04:13:30 +08:00
|
|
|
if (!this.resize_handles) {
|
|
|
|
this.setupFloatHandles();
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-03-10 04:52:10 +08:00
|
|
|
if (slot !== 'float') {
|
|
|
|
Panel.floating_panel_z_order.remove(this.id);
|
|
|
|
this.node.style.zIndex = '';
|
2022-03-16 23:20:27 +08:00
|
|
|
this.dispatchEvent('change_zindex', {zindex: null});
|
2022-03-10 04:52:10 +08:00
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
position_data.slot = slot;
|
2022-03-15 04:38:37 +08:00
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
this.update();
|
2022-02-23 01:12:27 +08:00
|
|
|
if (Panels[this.id]) {
|
|
|
|
updateInterface()
|
2022-03-13 05:57:54 +08:00
|
|
|
this.dispatchEvent('moved_to', {slot, ref_panel, before, previous_slot: this.previous_slot});
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
return this;
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-03-05 06:34:36 +08:00
|
|
|
update(dragging) {
|
2022-02-23 01:12:27 +08:00
|
|
|
let show = BARS.condition(this.condition);
|
2022-03-04 04:13:30 +08:00
|
|
|
let work_screen = document.querySelector('div#work_screen');
|
2022-02-23 01:12:27 +08:00
|
|
|
let center_screen = document.querySelector('div#center');
|
2020-07-16 15:32:59 +08:00
|
|
|
if (show) {
|
|
|
|
$(this.node).show()
|
2022-03-04 04:13:30 +08:00
|
|
|
if (this.slot == 'float') {
|
2022-03-07 21:37:19 +08:00
|
|
|
if (!dragging && work_screen.clientWidth) {
|
2022-03-05 06:34:36 +08:00
|
|
|
this.position_data.float_position[0] = Math.clamp(this.position_data.float_position[0], 0, work_screen.clientWidth - this.width);
|
|
|
|
this.position_data.float_position[1] = Math.clamp(this.position_data.float_position[1], 0, work_screen.clientHeight - this.height);
|
2022-03-11 04:24:33 +08:00
|
|
|
this.position_data.float_size[0] = Math.clamp(this.position_data.float_size[0], 200, work_screen.clientWidth - this.position_data.float_position[0]);
|
|
|
|
this.position_data.float_size[1] = Math.clamp(this.position_data.float_size[1], 86, work_screen.clientHeight - this.position_data.float_position[1]);
|
2022-03-05 06:34:36 +08:00
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
this.node.style.left = this.position_data.float_position[0] + 'px';
|
|
|
|
this.node.style.top = this.position_data.float_position[1] + 'px';
|
|
|
|
this.width = this.position_data.float_size[0];
|
|
|
|
this.height = this.position_data.float_size[1];
|
|
|
|
if (this.folded) this.height = this.handle.clientHeight;
|
|
|
|
this.node.style.width = this.width + 'px';
|
|
|
|
this.node.style.height = this.height + 'px';
|
|
|
|
} else {
|
|
|
|
this.node.style.width = this.node.style.height = this.node.style.left = this.node.style.top = null;
|
|
|
|
}
|
2021-10-15 18:15:11 +08:00
|
|
|
if (Blockbench.isMobile) {
|
|
|
|
this.width = this.node.clientWidth;
|
2022-02-23 01:12:27 +08:00
|
|
|
} else if (this.slot == 'left_bar') {
|
2020-07-16 15:32:59 +08:00
|
|
|
this.width = Interface.data.left_bar_width
|
2022-02-23 01:12:27 +08:00
|
|
|
} else if (this.slot == 'right_bar') {
|
2020-07-16 15:32:59 +08:00
|
|
|
this.width = Interface.data.right_bar_width
|
|
|
|
}
|
2022-02-23 01:12:27 +08:00
|
|
|
if (this.slot == 'top' || this.slot == 'bottom') {
|
2022-03-07 21:37:19 +08:00
|
|
|
|
|
|
|
if (Blockbench.isMobile && Blockbench.isLandscape) {
|
|
|
|
this.height = center_screen.clientHeight;
|
|
|
|
this.width = Math.clamp(this.position_data.height, 30, center_screen.clientWidth);
|
|
|
|
if (this.folded) this.width = 72;
|
|
|
|
} else {
|
2022-03-11 01:12:59 +08:00
|
|
|
let opposite_panel = this.slot == 'top' ? Interface.getBottomPanel() : Interface.getTopPanel();
|
|
|
|
this.height = Math.clamp(this.position_data.height, 30, center_screen.clientHeight - (opposite_panel ? opposite_panel.height : 0));
|
2022-03-07 21:37:19 +08:00
|
|
|
if (this.folded) this.height = this.handle.clientHeight;
|
|
|
|
this.width = Interface.work_screen.clientWidth - Interface.left_bar_width - Interface.right_bar_width;
|
|
|
|
}
|
|
|
|
this.node.style.width = this.width + 'px';
|
2022-02-23 01:12:27 +08:00
|
|
|
this.node.style.height = this.height + 'px';
|
|
|
|
}
|
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
if (Panels[this.id] && this.onResize) this.onResize()
|
2020-07-16 15:32:59 +08:00
|
|
|
} else {
|
|
|
|
$(this.node).hide()
|
|
|
|
}
|
2022-03-13 05:57:54 +08:00
|
|
|
this.dispatchEvent('update', {show});
|
2022-02-23 01:12:27 +08:00
|
|
|
localStorage.setItem('interface_data', JSON.stringify(Interface.data))
|
2022-03-04 04:13:30 +08:00
|
|
|
return this;
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
2022-03-13 05:57:54 +08:00
|
|
|
//Events
|
|
|
|
dispatchEvent(event_name, data) {
|
|
|
|
var list = this.events[event_name]
|
|
|
|
if (!list) return;
|
|
|
|
for (var i = 0; i < list.length; i++) {
|
|
|
|
if (typeof list[i] === 'function') {
|
|
|
|
list[i](data)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
on(event_name, cb) {
|
|
|
|
if (!this.events[event_name]) {
|
|
|
|
this.events[event_name] = []
|
|
|
|
}
|
|
|
|
this.events[event_name].safePush(cb)
|
|
|
|
}
|
|
|
|
removeListener(event_name, cb) {
|
|
|
|
if (this.events[event_name]) {
|
|
|
|
this.events[event_name].remove(cb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//Delete
|
2020-07-16 15:32:59 +08:00
|
|
|
delete() {
|
2022-03-11 01:34:00 +08:00
|
|
|
delete Panels[this.id];
|
2020-11-28 06:23:16 +08:00
|
|
|
this.node.remove()
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
Panel.floating_panel_z_order = [];
|
2020-07-16 15:32:59 +08:00
|
|
|
|
|
|
|
|
|
|
|
function setupPanels() {
|
2020-09-26 23:56:14 +08:00
|
|
|
Interface.panel_definers.forEach((definer) => {
|
|
|
|
if (typeof definer === 'function') {
|
|
|
|
definer()
|
2020-07-16 15:32:59 +08:00
|
|
|
}
|
|
|
|
})
|
2022-03-04 04:13:30 +08:00
|
|
|
updateSidebarOrder();
|
2022-02-23 01:12:27 +08:00
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
function updateInterfacePanels() {
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
if (!Blockbench.isMobile) {
|
|
|
|
$('.sidebar#left_bar').css('display', Prop.show_left_bar ? 'flex' : 'none');
|
|
|
|
$('.sidebar#right_bar').css('display', Prop.show_right_bar ? 'flex' : 'none');
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-03-11 01:12:59 +08:00
|
|
|
Interface.work_screen.style.setProperty(
|
2022-02-23 01:12:27 +08:00
|
|
|
'grid-template-columns',
|
|
|
|
Interface.data.left_bar_width+'px auto '+ Interface.data.right_bar_width +'px'
|
|
|
|
)
|
|
|
|
for (var key in Interface.Panels) {
|
2022-03-11 01:34:00 +08:00
|
|
|
var panel = Panels[key]
|
2022-02-23 01:12:27 +08:00
|
|
|
panel.update()
|
|
|
|
}
|
|
|
|
var left_width = $('.sidebar#left_bar > .panel:visible').length ? Interface.left_bar_width : 0;
|
|
|
|
var right_width = $('.sidebar#right_bar > .panel:visible').length ? Interface.right_bar_width : 0;
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-02-23 01:12:27 +08:00
|
|
|
if (!left_width || !right_width) {
|
2022-03-11 01:12:59 +08:00
|
|
|
Interface.work_screen.style.setProperty(
|
2022-02-23 01:12:27 +08:00
|
|
|
'grid-template-columns',
|
|
|
|
left_width+'px auto '+ right_width +'px'
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-03-11 01:12:59 +08:00
|
|
|
Interface.preview.style.visibility = Interface.preview.clientHeight > 80 ? 'visible' : 'hidden';
|
2022-02-23 01:12:27 +08:00
|
|
|
$('.quad_canvas_wrapper.qcw_x').css('width', Interface.data.quad_view_x+'%')
|
|
|
|
$('.quad_canvas_wrapper.qcw_y').css('height', Interface.data.quad_view_y+'%')
|
|
|
|
$('.quad_canvas_wrapper:not(.qcw_x)').css('width', (100-Interface.data.quad_view_x)+'%')
|
|
|
|
$('.quad_canvas_wrapper:not(.qcw_y)').css('height', (100-Interface.data.quad_view_y)+'%')
|
|
|
|
for (var key in Interface.Resizers) {
|
|
|
|
var resizer = Interface.Resizers[key]
|
|
|
|
resizer.update()
|
|
|
|
}
|
|
|
|
}
|
2020-07-16 15:32:59 +08:00
|
|
|
|
2022-03-04 04:13:30 +08:00
|
|
|
function updateSidebarOrder() {
|
|
|
|
['left_bar', 'right_bar'].forEach(bar => {
|
|
|
|
let bar_node = document.querySelector(`.sidebar#${bar}`);
|
|
|
|
|
|
|
|
bar_node.childNodes.forEach(panel_node => panel_node.remove());
|
|
|
|
|
|
|
|
Interface.data[bar].forEach(panel_id => {
|
|
|
|
let panel = Panels[panel_id];
|
2022-03-09 02:23:30 +08:00
|
|
|
if (panel && panel.slot == bar) {
|
|
|
|
bar_node.append(panel.node);
|
|
|
|
}
|
2022-03-04 04:13:30 +08:00
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
2020-09-22 05:23:42 +08:00
|
|
|
|
2020-07-16 15:32:59 +08:00
|
|
|
function setActivePanel(panel) {
|
|
|
|
Prop.active_panel = panel
|
|
|
|
}
|
|
|
|
|
2022-03-07 21:37:19 +08:00
|
|
|
function setupMobilePanelSelector() {
|
|
|
|
if (Blockbench.isMobile) {
|
|
|
|
Interface.PanelSelectorVue = new Vue({
|
|
|
|
el: '#panel_selector_bar',
|
|
|
|
data: {
|
|
|
|
all_panels: Interface.Panels,
|
|
|
|
selected: null,
|
|
|
|
modifiers: Pressing.overrides
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
panels() {
|
|
|
|
let arr = [];
|
|
|
|
for (var id in this.all_panels) {
|
|
|
|
let panel = this.all_panels[id];
|
|
|
|
if (Condition(panel.condition) && Condition(panel.display_condition)) {
|
|
|
|
arr.push(panel);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
},
|
|
|
|
select(panel) {
|
|
|
|
this.selected = panel && panel.id;
|
|
|
|
if (panel) {
|
|
|
|
panel.moveTo('bottom');
|
|
|
|
} else {
|
|
|
|
let other_panel = Interface.getBottomPanel();
|
|
|
|
if (other_panel) {
|
|
|
|
$(other_panel.node).detach();
|
|
|
|
}
|
|
|
|
resizeWindow();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
openKeyboardMenu(event) {
|
|
|
|
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(this.$refs.mobile_keyboard_menu)
|
|
|
|
},
|
|
|
|
Condition,
|
|
|
|
getIconNode: Blockbench.getIconNode
|
|
|
|
},
|
|
|
|
template: `
|
|
|
|
<div id="panel_selector_bar">
|
|
|
|
<div class="panel_selector" :class="{selected: selected == panel.id}" v-for="panel in panels()" v-if="Condition(panel.condition)" @click="select(panel)">
|
|
|
|
<div class="icon_wrapper" v-html="getIconNode(panel.icon).outerHTML"></div>
|
|
|
|
</div>
|
|
|
|
<div id="mobile_keyboard_menu" @click="openKeyboardMenu($event)" ref="mobile_keyboard_menu" :class="{enabled: modifiers.ctrl || modifiers.shift || modifiers.alt}">
|
|
|
|
<i class="material-icons">keyboard</i>
|
|
|
|
</div>
|
|
|
|
</div>`
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|