blockbench/js/io/project.js

793 lines
21 KiB
JavaScript
Raw Normal View History

2020-07-16 09:32:59 +02:00
class ModelProject {
2021-05-05 13:54:41 +02:00
constructor(options = {}) {
2020-07-16 09:32:59 +02:00
for (var key in ModelProject.properties) {
ModelProject.properties[key].reset(this, true);
2020-07-16 09:32:59 +02:00
}
2021-05-05 13:54:41 +02:00
this.uuid = guid();
this.selected = false;
this.locked = false;
2021-08-20 20:35:13 +02:00
this.thumbnail = '';
2020-07-16 09:32:59 +02:00
this._box_uv = options.format ? options.format.box_uv : false;
2020-07-16 09:32:59 +02:00
this._texture_width = 16;
this._texture_height = 16;
2021-05-05 13:54:41 +02:00
this._name = '';
this.saved = true;
this.save_path = '';
this.export_path = '';
this.added_models = 0;
2021-05-05 13:54:41 +02:00
this.undo = new UndoSystem();
if (isApp) this.BedrockEntityManager = new BedrockEntityManager(this);
this.format = options.format instanceof ModelFormat ? options.format : Formats.free;
2021-07-10 22:22:02 +02:00
this.mode = 'edit';
this.view_mode = 'textured';
this.display_uv = settings.show_only_selected_uv.value ? 'selected_faces' :'selected_elements';
this.previews = {};
this.EditSession = null;
// Data
this.elements = [];
this.groups = [];
this.selected_elements = [];
this.selected_group = null;
2021-08-02 22:55:30 +02:00
this.selected_vertices = {};
2021-08-30 10:30:46 +02:00
this.selected_faces = [];
this.textures = [];
this.selected_texture = null;
this.outliner = [];
2021-07-10 22:22:02 +02:00
this.animations = [];
this.timeline_animators = [];
this.display_settings = {};
2021-05-05 13:54:41 +02:00
ModelProject.all.push(this);
ProjectData[this.uuid] = {
model_3d: new THREE.Object3D(),
materials: {},
nodes_3d: {}
}
2020-07-16 09:32:59 +02:00
}
extend() {
for (var key in ModelProject.properties) {
ModelProject.properties[key].merge(this, object)
}
}
get box_uv() {return Project._box_uv}
2020-04-25 20:25:07 +02:00
set box_uv(v) {
if (Project._box_uv != v) {
Project._box_uv = v;
switchBoxUV(v);
}
2020-07-16 09:32:59 +02:00
}
get texture_width() {return this._texture_width}
get texture_height() {return this._texture_height}
2020-04-25 20:25:07 +02:00
set texture_width(n) {
n = parseInt(n)||16
2021-08-19 16:01:50 +02:00
if (this.selected && n != this._texture_width) {
Vue.nextTick(updateProjectResolution);
}
2020-07-16 09:32:59 +02:00
this._texture_width = n;
}
2021-05-05 13:54:41 +02:00
get optional_box_uv() {
return Format.optional_box_uv;
}
2020-04-25 20:25:07 +02:00
set texture_height(n) {
n = parseInt(n)||16
2021-08-19 16:01:50 +02:00
if (this.selected && n != this._texture_height) {
Vue.nextTick(updateProjectResolution);
}
2020-07-16 09:32:59 +02:00
this._texture_height = n;
}
2021-05-05 13:54:41 +02:00
get name() {
return this._name;
}
set name(name) {
this._name = name;
2021-07-10 22:22:02 +02:00
if (Project == this) {
setProjectTitle(this._name);
}
2020-04-25 20:25:07 +02:00
}
get model_3d() {
return ProjectData[this.uuid].model_3d;
}
get materials() {
return ProjectData[this.uuid].materials;
}
get nodes_3d() {
return ProjectData[this.uuid].nodes_3d;
}
getDisplayName() {
return this.name || this.geometry_name || this.format.name;
}
2021-05-05 13:54:41 +02:00
openSettings() {
if (this.selected) BarItems.project_window.click();
2021-05-05 13:54:41 +02:00
}
whenNextOpen(callback) {
if (Project == this) {
callback();
} else {
if (!this.on_next_upen) this.on_next_upen = [];
this.on_next_upen.push(callback);
}
}
2021-05-05 13:54:41 +02:00
select() {
if (this === Project) return true;
if (this.locked || Project.locked) return false;
2021-05-05 13:54:41 +02:00
if (Project) {
Project.unselect()
} else {
Interface.tab_bar.new_tab.visible = false;
2021-05-05 13:54:41 +02:00
}
Project = this;
Undo = this.undo;
this.selected = true;
this.format.select();
BarItems.view_mode.set(this.view_mode);
// Setup Data
OutlinerNode.uuids = {};
this.elements.forEach(el => {
OutlinerNode.uuids[el.uuid] = el;
})
this.groups.forEach(group => {
OutlinerNode.uuids[group.uuid] = group;
})
Outliner.root = this.outliner;
Interface.Panels.outliner.inside_vue.root = this.outliner;
2021-08-17 22:02:23 +02:00
UVEditor.vue.elements = this.selected_elements;
2021-08-25 22:00:08 +02:00
UVEditor.vue.all_elements = this.elements;
2021-08-17 22:02:23 +02:00
UVEditor.vue.selected_vertices = this.selected_vertices;
UVEditor.vue.selected_faces = this.selected_faces;
2021-08-19 15:18:01 +02:00
UVEditor.vue.box_uv = this.box_uv;
UVEditor.vue.display_uv = this.display_uv;
2021-08-17 22:02:23 +02:00
2021-07-10 22:22:02 +02:00
Interface.Panels.textures.inside_vue.textures = Texture.all;
scene.add(this.model_3d);
2021-07-10 22:22:02 +02:00
Interface.Panels.animations.inside_vue.animations = this.animations;
Animation.selected = null;
let selected_anim = this.animations.find(anim => anim.selected);
if (selected_anim) selected_anim.select();
Timeline.animators = this.timeline_animators;
2021-07-10 22:22:02 +02:00
Timeline.vue.animators = this.timeline_animators;
Interface.Panels.variable_placeholders.inside_vue.text = this.variable_placeholders.toString();
Modes.options[this.mode].select();
BarItems.lock_motion_trail.value = !!Project.motion_trail_lock;
BarItems.lock_motion_trail.updateEnabledState();
Preview.all.forEach(preview => {
let data = this.previews[preview.id];
if (data) {
preview.camera.position.fromArray(data.position);
preview.controls.target.fromArray(data.target);
preview.setProjectionMode(data.orthographic);
if (data.zoom) preview.camOrtho.zoom = data.zoom;
if (data.angle) preview.setLockedAngle(data.angle);
} else if (preview.default_angle !== undefined) {
setTimeout(() => preview.loadAnglePreset(preview.default_angle), 0);
}
})
if (this.EditSession) {
Interface.Panels.chat.inside_vue.chat_history = this.EditSession.chat_history;
this.EditSession.catchUp();
}
Blockbench.dispatchEvent('select_project', {project: this});
2021-07-10 22:22:02 +02:00
if (Preview.selected) Preview.selected.occupyTransformer();
2021-07-10 22:22:02 +02:00
setProjectTitle(this.name);
setStartScreen(!Project);
updateInterface();
2021-08-19 15:18:01 +02:00
updateProjectResolution();
2021-07-10 22:22:02 +02:00
Vue.nextTick(() => {
loadTextureDraggable();
if (this.on_next_upen instanceof Array) {
this.on_next_upen.forEach(callback => callback());
delete this.on_next_upen;
}
2021-07-10 22:22:02 +02:00
})
return true;
2021-05-05 13:54:41 +02:00
}
unselect() {
2021-08-20 20:35:13 +02:00
this.thumbnail = Preview.selected.canvas.toDataURL();
Interface.tab_bar.last_opened_project = this.uuid;
2021-08-20 20:35:13 +02:00
if (Format && typeof Format.onDeactivation == 'function') {
Format.onDeactivation()
}
Preview.all.forEach(preview => {
this.previews[preview.id] = {
position: preview.camera.position.toArray(),
target: preview.controls.target.toArray(),
orthographic: preview.isOrtho,
zoom: preview.camOrtho.zoom,
angle: preview.angle,
}
})
if (TextureAnimator.isPlaying) TextureAnimator.stop();
2021-05-05 13:54:41 +02:00
this.selected = false;
2021-07-10 22:22:02 +02:00
Painter.current = {};
scene.remove(this.model_3d);
2021-07-10 22:22:02 +02:00
OutlinerNode.uuids = {};
Format = 0;
Project = 0;
Undo = 0;
OutlinerNode.uuids = {};
Outliner.root = [];
Blockbench.dispatchEvent('unselect_project', {project: this});
2021-05-05 13:54:41 +02:00
}
async close(force) {
if (this.locked) return false;
let last_selected = Project;
try {
let result = this.select();
if (result === false) return false;
} catch (err) {
console.error(err);
}
2021-05-05 13:54:41 +02:00
async function saveWarning() {
await new Promise(resolve => setTimeout(resolve, 4));
if (Project.saved) {
return true;
} else {
if (isApp) {
var answer = electron.dialog.showMessageBoxSync(currentwindow, {
type: 'question',
buttons: [tl('dialog.save'), tl('dialog.discard'), tl('dialog.cancel')],
title: 'Blockbench',
message: tl('message.close_warning.message'),
noLink: true
})
if (answer === 0) {
BarItems.save_project.trigger();
}
return answer !== 2;
} else {
var answer = confirm(tl('message.close_warning.web'))
return answer;
}
}
}
if (force || await saveWarning()) {
2021-05-05 13:54:41 +02:00
if (isApp) await updateRecentProjectThumbnail();
Blockbench.dispatchEvent('close_project');
2021-07-10 22:22:02 +02:00
this.unselect();
Texture.all.forEach(tex => tex.stopWatcher());
2021-05-05 13:54:41 +02:00
let index = ModelProject.all.indexOf(this);
2021-07-10 22:22:02 +02:00
ModelProject.all.remove(this);
delete ProjectData[this.uuid];
2021-07-10 22:22:02 +02:00
Project = 0;
if (last_selected && last_selected !== this) {
last_selected.select();
} else if (ModelProject.all.length) {
ModelProject.all[Math.clamp(index, 0, ModelProject.all.length-1)].select();
2021-07-10 22:22:02 +02:00
} else {
Interface.tab_bar.new_tab.visible = true;
Interface.tab_bar.new_tab.select();
setStartScreen(true);
}
2021-05-05 13:54:41 +02:00
return true;
} else {
return false;
}
}
2020-04-25 20:25:07 +02:00
}
2020-07-16 09:32:59 +02:00
new Property(ModelProject, 'string', 'name', {
label: 'dialog.project.name'
});
new Property(ModelProject, 'string', 'parent', {
label: 'dialog.project.parent',
condition: {formats: ['java_block']
}});
new Property(ModelProject, 'string', 'geometry_name', {
label: 'dialog.project.geoname',
condition: () => Format.bone_rig
});
new Property(ModelProject, 'string', 'modded_entity_version', {
label: 'dialog.project.modded_entity_version',
default: '1.17',
2020-07-16 09:32:59 +02:00
condition: {formats: ['modded_entity']},
options() {
let options = {}
for (var key in Codecs.modded_entity.templates) {
if (Codecs.modded_entity.templates[key] instanceof Function == false) {
options[key] = Codecs.modded_entity.templates[key].name;
}
}
return options;
}
});
new Property(ModelProject, 'boolean', 'ambientocclusion', {
label: 'dialog.project.ao',
default: true,
condition: {formats: ['java_block']}
});
new Property(ModelProject, 'boolean', 'front_gui_light', {
exposed: false,
condition: () => Format.display_mode});
new Property(ModelProject, 'vector', 'visible_box', {
exposed: false,
default: [1, 1, 0]
});
2021-07-10 22:22:02 +02:00
new Property(ModelProject, 'string', 'variable_placeholders', {
exposed: false,
});
new Property(ModelProject, 'number', 'shadow_size', {
label: 'dialog.project.shadow_size',
condition: {formats: ['optifine_entity']},
default: 1
});
2020-07-16 09:32:59 +02:00
2021-05-05 13:54:41 +02:00
ModelProject.all = [];
let Project = 0;
2020-04-25 20:25:07 +02:00
let ProjectData = {};
// Setup ModelProject for loaded project
function setupProject(format) {
if (typeof format == 'string' && Formats[format]) format = Formats[format];
new ModelProject({format}).select();
Modes.options.edit.select();
Blockbench.dispatchEvent('setup_project');
return true;
}
// Setup brand new project
2021-05-05 13:54:41 +02:00
function newProject(format) {
if (typeof format == 'string' && Formats[format]) format = Formats[format];
2021-05-05 13:54:41 +02:00
new ModelProject({format}).select();
Modes.options.edit.select();
Blockbench.dispatchEvent('new_project');
return true;
2020-04-25 20:25:07 +02:00
}
2021-01-23 23:24:14 +01:00
// Resolution
function setProjectResolution(width, height, modify_uv) {
if (Project.texture_width / width != Project.texture_width / height) {
modify_uv = false;
}
Undo.initEdit({uv_mode: true, elements: Cube.all, uv_only: true})
let old_res = {
x: Project.texture_width,
y: Project.texture_height
}
Project.texture_width = width;
Project.texture_height = height;
if (modify_uv) {
var multiplier = [
Project.texture_width/old_res.x,
Project.texture_height/old_res.y
]
2021-08-20 20:02:28 +02:00
function shiftElement(element, axis) {
if (!element.faces) return;
if (element instanceof Mesh) {
for (let key in element.faces) {
let face = element.faces[key];
face.vertices.forEach(vertex_key => {
if (face.uv[vertex_key]) {
face.uv[vertex_key][axis] *= multiplier[axis];
}
})
}
} else if (Project.box_uv) {
element.uv_offset[axis] *= multiplier[axis];
2021-01-23 23:24:14 +01:00
} else {
2021-08-20 20:02:28 +02:00
for (let face in element.faces) {
let {uv} = element.faces[face];
2021-01-23 23:24:14 +01:00
uv[axis] *= multiplier[axis];
uv[axis+2] *= multiplier[axis];
}
}
}
if (old_res.x != Project.texture_width && Math.areMultiples(old_res.x, Project.texture_width)) {
2021-08-20 20:02:28 +02:00
Outliner.elements.forEach(element => shiftElement(element, 0));
2021-01-23 23:24:14 +01:00
}
if (old_res.y != Project.texture_height && Math.areMultiples(old_res.x, Project.texture_width)) {
2021-08-20 20:02:28 +02:00
Outliner.elements.forEach(element => shiftElement(element, 1));
2021-01-23 23:24:14 +01:00
}
}
Undo.finishEdit('Changed project resolution')
Canvas.updateAllUVs()
if (selected.length) {
2021-08-17 22:02:23 +02:00
UVEditor.loadData()
2021-01-23 23:24:14 +01:00
}
}
function updateProjectResolution() {
2021-08-17 22:02:23 +02:00
if (Interface.Panels.uv) {
UVEditor.vue.project_resolution.replace([Project.texture_width, Project.texture_height]);
UVEditor.vue.updateSize()
2021-08-17 22:02:23 +02:00
}
if (Texture.selected) {
// Update animated textures
Texture.selected.height++;
Texture.selected.height--;
}
Blockbench.dispatchEvent('update_project_resolution', {project: Project});
2021-01-23 23:24:14 +01:00
}
2021-07-07 13:08:56 +02:00
function setStartScreen(state) {
document.getElementById('start_screen').style.display = state ? 'block' : 'none';
document.getElementById('work_screen').style.display = state ? 'none' : 'grid';
}
2021-05-05 13:54:41 +02:00
onVueSetup(() => {
const new_tab = {
name: tl('projects.new_tab'),
saved: true,
selected: true,
uuid: guid(),
visible: true,
is_new_tab: true,
getDisplayName() {return this.name},
close: () => {
if (ModelProject.all.length) {
Interface.tab_bar.new_tab.visible = false;
let project = ModelProject.all.find(project => project.uuid == Interface.tab_bar.last_opened_project) ||
ModelProject.all.last();
if (project) project.select();
} else {
window.close();
}
},
select() {
if (Project) {
Project.unselect()
}
Project = 0;
Interface.tab_bar.new_tab.selected = true;
setProjectTitle(tl('projects.new_tab'));
},
openSettings() {}
}
2021-05-05 13:54:41 +02:00
Interface.tab_bar = new Vue({
el: '#tab_bar',
data: {
projects: ModelProject.all,
2021-05-08 16:52:49 +02:00
drag_target_index: null,
drag_position_index: null,
2021-07-20 12:07:47 +02:00
close_tab_label: tl('projects.close_tab'),
search_tabs_label: tl('generic.search'),
last_opened_project: '',
new_tab
},
computed: {
tabs() {
let tabs = this.projects.slice();
if (this.new_tab.visible) {
tabs.push(this.new_tab);
}
return tabs;
}
},
methods: {
openNewTab() {
if (Project.locked) return;
this.last_opened_project = Project.uuid;
this.new_tab.visible = true;
this.new_tab.select();
2021-07-07 13:08:56 +02:00
setStartScreen(true);
2021-05-08 16:52:49 +02:00
},
searchTabs() {
ActionControl.select('tab: ');
},
2021-07-20 12:07:47 +02:00
mouseDown(tab, e1) {
2021-05-08 16:52:49 +02:00
convertTouchEvent(e1);
e1.preventDefault();
2021-05-08 16:52:49 +02:00
2021-08-20 20:35:13 +02:00
if (this.thumbnail) {
this.thumbnail.remove();
delete this.thumbnail;
}
if (e1.button == 1) return;
2021-08-20 20:35:13 +02:00
2021-05-08 16:52:49 +02:00
let scope = this;
let active = false;
let timeout;
let last_event = e1;
let tab_node = e1.target;
if (!tab_node.classList.contains('project_tab') || ModelProject.all.indexOf(tab) < 0) return;
2021-07-20 12:07:47 +02:00
tab.select();
2021-05-08 16:52:49 +02:00
let activate = () => {
this.drag_target_index = ModelProject.all.indexOf(tab);
this.drag_position_index = 0;
if (open_menu) open_menu.hide();
active = true;
}
function move(e2) {
convertTouchEvent(e2);
let offset = e2.clientX - e1.clientX;
if (!active) {
let distance = Math.abs(offset);
if (Blockbench.isTouch) {
if (distance > 14 && timeout) {
clearTimeout(timeout);
timeout = null;
} else {
document.getElementById('tab_bar').scrollLeft += last_event.clientX - e2.clientX;
}
} else if (distance > 5) {
activate();
}
} else {
if (e2) e2.preventDefault();
tab_node.style.left = `${offset}px`;
let index_offset = Math.trunc((e2.clientX - e1.clientX) / tab_node.clientWidth);
scope.drag_position_index = scope.drag_target_index + index_offset;
}
last_event = e2;
}
function off(e2) {
let {drag_target_index} = scope;
removeEventListeners(document, 'mousemove touchmove', move);
removeEventListeners(document, 'mouseup touchend', off);
tab_node.style.left = null;
scope.drag_target_index = null;
scope.drag_position_index = null;
if (Blockbench.isTouch) clearTimeout(timeout);
if (active && !open_menu) {
convertTouchEvent(e2);
let index_offset = Math.trunc((e2.clientX - e1.clientX) / tab_node.clientWidth);
if (index_offset) {
ModelProject.all.splice(drag_target_index, 1);
ModelProject.all.splice(drag_target_index + index_offset, 0, tab);
}
}
}
if (Blockbench.isTouch) {
timeout = setTimeout(() => {
active = true;
move(e1);
}, 320)
}
addEventListeners(document, 'mousemove touchmove', move, {passive: false});
addEventListeners(document, 'mouseup touchend', off, {passive: false});
},
2021-07-10 22:22:02 +02:00
selectProject(project, event) {
if (!event.target.classList.contains('project_tab_close_button')) {
project.select();
}
},
2021-05-08 16:52:49 +02:00
mouseUp(tab, e1) {
if (e1.button === 1) {
tab.close()
}
2021-08-20 20:35:13 +02:00
},
mouseEnter(project, event) {
if (project.thumbnail && !project.selected) {
if (this.thumbnail_timeout) {
clearTimeout(this.thumbnail_timeout);
delete this.thumbnail_timeout;
}
if (!this.thumbnail) {
this.thumbnail = new Image();
document.body.append(this.thumbnail);
}
let img = this.thumbnail;
2021-08-20 20:35:13 +02:00
img.src = project.thumbnail;
img.attributes.width = '240px';
img.className = 'project_thumbnail';
let offset = $(event.target).offset();
img.style.left = (offset.left) + 'px';
img.style.top = (offset.top + event.target.clientHeight+2) + 'px';
2021-08-20 20:35:13 +02:00
}
},
mouseLeave() {
if (this.thumbnail) {
this.thumbnail_timeout = setTimeout(() => {
if (this.thumbnail) this.thumbnail.remove();
delete this.thumbnail;
delete this.thumbnail_timeout;
}, 80)
2021-08-20 20:35:13 +02:00
}
}
2021-05-05 13:54:41 +02:00
}
})
})
2020-04-25 20:25:07 +02:00
BARS.defineActions(function() {
new Action('project_window', {
icon: 'featured_play_list',
category: 'file',
condition: () => Format,
click: function () {
2020-07-16 09:32:59 +02:00
let form = {
format: {type: 'info', label: 'data.format', text: Format.name||'unknown'}
}
for (var key in ModelProject.properties) {
let property = ModelProject.properties[key];
if (property.exposed == false || !Condition(property.condition)) continue;
let entry = form[property.name] = {
label: property.label,
2020-12-30 22:27:12 +01:00
description: property.description,
2020-07-16 09:32:59 +02:00
value: Project[property.name],
type: property.type
}
if (property.type == 'boolean') entry.type = 'checkbox';
if (property.type == 'string') entry.type = 'text';
if (property.options) {
entry.options = typeof property.options == 'function' ? property.options() : property.options;
entry.type = 'select';
2020-04-25 20:25:07 +02:00
}
}
2020-07-16 09:32:59 +02:00
form.uv_mode = {
label: 'dialog.project.uv_mode',
type: 'select',
condition: Format.optional_box_uv,
options: {
face_uv: 'dialog.project.uv_mode.face_uv',
box_uv: 'dialog.project.uv_mode.box_uv',
},
value: Project.box_uv ? 'box_uv' : 'face_uv',
2020-07-16 09:32:59 +02:00
};
form.texture_size = {
label: 'dialog.project.texture_size',
type: 'vector',
dimensions: 2,
value: [Project.texture_width, Project.texture_height],
2020-07-16 09:32:59 +02:00
min: 1
};
2020-04-25 20:25:07 +02:00
var dialog = new Dialog({
id: 'project',
title: 'dialog.project.title',
2020-07-16 09:32:59 +02:00
width: 500,
form,
2020-04-25 20:25:07 +02:00
onConfirm: function(formResult) {
var save;
let box_uv = formResult.uv_mode == 'box_uv';
2020-12-02 11:32:08 +01:00
let texture_width = Math.clamp(formResult.texture_size[0], 1, Infinity);
let texture_height = Math.clamp(formResult.texture_size[1], 1, Infinity);
if (Project.box_uv != box_uv ||
Project.texture_width != texture_width ||
Project.texture_height != texture_height
2020-04-25 20:25:07 +02:00
) {
if (!Project.box_uv && !box_uv
&& (Project.texture_width != texture_width
|| Project.texture_height != texture_height)
2020-04-25 20:25:07 +02:00
) {
save = Undo.initEdit({uv_only: true, elements: [...Cube.all, ...Mesh.all], uv_mode: true})
2020-04-25 20:25:07 +02:00
Cube.all.forEach(cube => {
for (var key in cube.faces) {
var uv = cube.faces[key].uv;
uv[0] *= texture_width / Project.texture_width;
uv[2] *= texture_width / Project.texture_width;
uv[1] *= texture_height / Project.texture_height;
uv[3] *= texture_height / Project.texture_height;
2020-04-25 20:25:07 +02:00
}
})
Mesh.all.forEach(mesh => {
for (var key in mesh.faces) {
var uv = mesh.faces[key].uv;
for (let vkey in uv) {
uv[vkey][0] *= texture_width / Project.texture_width;
uv[vkey][1] *= texture_height / Project.texture_height;
}
}
})
2020-04-25 20:25:07 +02:00
} else {
save = Undo.initEdit({uv_mode: true})
}
Project.texture_width = texture_width;
Project.texture_height = texture_height;
2020-04-25 20:25:07 +02:00
if (Format.optional_box_uv) Project.box_uv = box_uv;
2020-04-25 20:25:07 +02:00
Canvas.updateAllUVs()
updateSelection()
}
2020-07-16 09:32:59 +02:00
for (var key in ModelProject.properties) {
ModelProject.properties[key].merge(Project, formResult);
}
2020-04-25 20:25:07 +02:00
if (save) {
2021-06-06 09:28:22 +02:00
Undo.finishEdit('Change project UV settings')
2020-04-25 20:25:07 +02:00
}
Blockbench.dispatchEvent('update_project_settings', formResult);
2020-04-25 20:25:07 +02:00
BARS.updateConditions()
if (Project.EditSession) {
Project.EditSession.sendAll('change_project_meta', JSON.stringify(Project));
2020-04-25 20:25:07 +02:00
}
2020-04-25 20:25:07 +02:00
dialog.hide()
}
})
dialog.show()
}
})
new Action('close_project', {
icon: 'cancel_presentation',
category: 'file',
keybind: new Keybind({key: 'w', ctrl: true}),
condition: () => Project,
2020-04-25 20:25:07 +02:00
click: function () {
Project.close();
2020-04-25 20:25:07 +02:00
}
})
2020-04-25 20:25:07 +02:00
new Action('convert_project', {
icon: 'fas.fa-file-import',
category: 'file',
condition: () => Project && (!Project.EditSession || Project.EditSession.hosting),
2020-04-25 20:25:07 +02:00
click: function () {
var options = {};
for (var key in Formats) {
if (key !== Format.id && key !== 'skin') {
options[key] = Formats[key].name;
}
}
var dialog = new Dialog({
id: 'convert_project',
title: 'dialog.convert_project.title',
width: 540,
form: {
text: {type: 'info', text: 'dialog.convert_project.text'},
current: {type: 'info', label: 'dialog.convert_project.current_format', text: Format.name || '-'},
format: {
2020-04-25 20:25:07 +02:00
label: 'data.format',
type: 'select',
default: Format.id,
options,
},
},
onConfirm: function(formResult) {
var format = Formats[formResult.format]
if (format && format != Format) {
format.convertTo()
}
dialog.hide()
}
})
dialog.show()
}
})
})