blockbench/js/preview/preview.js

1836 lines
53 KiB
JavaScript
Raw Normal View History

2019-07-18 00:02:07 +08:00
var scene, main_preview, previews,
Sun, lights,
2019-12-16 03:04:31 +08:00
emptyMaterials,
outlines,
2019-07-18 00:02:07 +08:00
Transformer,
canvas_scenes,
display_scene, display_area, display_base;
var framespersecond = 0;
var display_mode = false;
var doRender = false;
var quad_previews = {};
const three_grid = new THREE.Object3D();
const rot_origin = new THREE.Object3D();
var gizmo_colors = {
r: new THREE.Color(0xfd3043),
g: new THREE.Color(0x26ec45),
2019-08-18 00:26:14 +08:00
b: new THREE.Color(0x2d5ee8),
2019-12-16 03:04:31 +08:00
grid: new THREE.Color(0x495061),
wire: new THREE.Color(0x576f82),
outline: new THREE.Color(0x3e90ff)
}
2020-03-05 03:56:17 +08:00
const DefaultCameraPresets = [
{
name: 'menu.preview.angle.initial',
projection: 'perspective',
position: [-40, 32, -40],
target: [0, 8, 0],
default: true
},
{
name: 'direction.top',
projection: 'orthographic',
color: 'y',
position: [0, 64, 0],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 0,
default: true
},
{
name: 'direction.bottom',
projection: 'orthographic',
color: 'y',
position: [0, -64, 0],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 1,
default: true
},
{
name: 'direction.south',
projection: 'orthographic',
color: 'z',
position: [0, 0, 64],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 2,
default: true
},
{
name: 'direction.north',
projection: 'orthographic',
color: 'z',
position: [0, 0, -64],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 3,
default: true
},
{
name: 'direction.east',
projection: 'orthographic',
color: 'x',
position: [64, 0, 0],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 4,
default: true
},
{
name: 'direction.west',
projection: 'orthographic',
color: 'x',
position: [-64, 0, 0],
target: [0, 0, 0],
zoom: 0.5,
locked_angle: 5,
default: true
}
]
2019-07-18 00:02:07 +08:00
class Preview {
constructor(data) {
var scope = this;
if (data && data.id) {
this.id = data.id
}
//Node
this.canvas = document.createElement('canvas')
this.canvas.preview = this;
this.canvas.className = 'preview';
this.height = 0;
this.width = 0;
//Cameras
this.isOrtho = false
2020-03-05 03:56:17 +08:00
this.angle = null;
2019-12-16 03:04:31 +08:00
this.camPers = new THREE.PerspectiveCamera(45, 16 / 9, 1, 30000)
2020-03-05 03:56:17 +08:00
this.camOrtho = new THREE.OrthographicCamera(-600, 600, -400, 400, -200, 20000);
2019-07-18 00:02:07 +08:00
this.camOrtho.backgroundHandle = [{n: false, a: 'x'}, {n: false, a: 'y'}]
this.camOrtho.axis = null
2020-03-05 03:56:17 +08:00
this.camOrtho.zoom = 0.5
2019-07-18 00:02:07 +08:00
this.camPers.preview = this.camOrtho.preview = this;
for (var i = 4; i <= 6; i++) {
this.camPers.layers.enable(i);
}
//Controls
this.controls = new THREE.OrbitControls(this.camPers, this);
this.controls.minDistance = 1;
2019-12-16 03:04:31 +08:00
this.controls.maxDistance = 3960;
2019-07-18 00:02:07 +08:00
this.controls.enableKeys = false;
2019-12-16 03:04:31 +08:00
this.controls.zoomSpeed = 1.5;
2020-01-24 01:53:36 +08:00
//Annotations
this.annotations = {};
this.updateAnnotations = function() {
for (var key in scope.annotations) {
var tag = scope.annotations[key];
2019-12-16 03:04:31 +08:00
if (tag.object.visible) {
2020-01-24 01:53:36 +08:00
var pos = tag.object.toScreenPosition(scope.camera, scope.canvas);
2019-12-16 03:04:31 +08:00
$(tag.node).css('left', pos.x+'px');
$(tag.node).css('top', pos.y+'px');
}
2020-01-24 01:53:36 +08:00
}
}
this.controls.onUpdate(() => setTimeout(() => {
scope.updateAnnotations();
}, 6))
this.addAnnotation = function(key, tag) {
scope.annotations[key] = tag;
$(tag.node).insertBefore(scope.canvas);
scope.updateAnnotations();
}
this.removeAnnotation = function(key) {
if (scope.annotations[key]) {
$(scope.annotations[key].node).detach();
delete scope.annotations[key];
}
2019-12-16 03:04:31 +08:00
}
2019-07-18 00:02:07 +08:00
2020-03-05 03:56:17 +08:00
this.camPers.position.fromArray(DefaultCameraPresets[0].position)
this.controls.target.fromArray(DefaultCameraPresets[0].target);
2019-07-18 00:02:07 +08:00
//Keybinds
this.controls.mouseButtons.ZOOM = undefined;
//Renderer
this.renderer = new THREE.WebGLRenderer({
canvas: this.canvas,
antialias: true,
alpha: true,
preserveDrawingBuffer: true
});
this.renderer.setClearColor( 0x000000, 0 )
this.renderer.setSize(500, 400);
this.loadBackground()
this.selection = {
2019-08-18 00:26:14 +08:00
box: $('<div id="selection_box", class="selection_rectangle"></div>')
2019-07-18 00:02:07 +08:00
}
this.raycaster = new THREE.Raycaster()
this.mouse = new THREE.Vector2();
2019-12-16 03:04:31 +08:00
addEventListeners(this.canvas, 'mousedown touchstart', function(event) { scope.click(event)}, { passive: false })
addEventListeners(this.canvas, 'mousemove touchmove', function(event) { scope.static_rclick = false}, false)
addEventListeners(this.canvas, 'mousemove', function(event) { scope.mousemove(event)}, false)
addEventListeners(this.canvas, 'mouseup touchend', function(event) { scope.showContextMenu(event)}, false)
addEventListeners(this.canvas, 'dblclick', function(event) {Toolbox.toggleTransforms(event)}, false)
addEventListeners(this.canvas, 'mouseenter touchstart', function(event) { scope.occupyTransformer(event)}, false)
2019-07-18 00:02:07 +08:00
Blockbench.addDragHandler('preview_'+this.id, {
extensions: ['jpg', 'jpeg', 'bmp', 'tiff', 'tif', 'gif'],
element: this.canvas,
readtype: 'image',
}, function(files) {
2019-12-16 03:04:31 +08:00
if (!scope.background.imgtag) {
scope.background.imgtag = new Image();
}
2019-07-18 00:02:07 +08:00
if (isApp) {
scope.background.image = files[0].path
} else {
scope.background.image = files[0].content
}
scope.loadBackground()
})
previews.push(this)
}
//Render
resize() {
if (!this.canvas.isConnected) return;
this.height = this.canvas.parentElement.clientHeight;
this.width = this.canvas.parentElement.clientWidth;
if (this.isOrtho === false) {
this.camPers.aspect = this.width / this.height
this.camPers.updateProjectionMatrix();
if (Transformer) {
Transformer.update()
}
} else {
this.camOrtho.right = this.width / 80
this.camOrtho.left = this.camOrtho.right*-1
this.camOrtho.top = this.height / 80
this.camOrtho.bottom = this.camOrtho.top*-1
this.camOrtho.updateProjectionMatrix();
}
this.renderer.setSize(this.width, this.height);
this.renderer.setPixelRatio(window.devicePixelRatio);
this.updateBackground()
return this;
}
raycast(event) {
2019-12-16 03:04:31 +08:00
convertTouchEvent(event);
2019-07-18 00:02:07 +08:00
var canvas_offset = $(this.canvas).offset()
2019-12-16 03:04:31 +08:00
this.mouse.x = ((event.clientX - canvas_offset.left) / this.width) * 2 - 1;
this.mouse.y = - ((event.clientY - canvas_offset.top) / this.height) * 2 + 1;
2020-03-05 03:56:17 +08:00
this.raycaster.setFromCamera( this.mouse, this.camera );
2019-07-18 00:02:07 +08:00
var objects = []
scene.traverse(function(s) {
if (s.isElement === true) {
objects.push(s)
}
})
if (Vertexsnap.vertexes.children.length) {
Vertexsnap.vertexes.children.forEach(function(s) {
if (s.isVertex === true) {
objects.push(s)
}
})
}
var intersects = this.raycaster.intersectObjects( objects );
if (intersects.length > 0) {
var intersect = intersects[0].object
if (intersect.isElement) {
this.controls.hasMoved = true
2019-08-01 06:01:47 +08:00
var obj = elements.findInArray('uuid', intersects[0].object.name)
2019-07-18 00:02:07 +08:00
switch (Math.floor( intersects[0].faceIndex / 2 )) {
case 5: var face = 'north'; break;
case 0: var face = 'east'; break;
case 4: var face = 'south'; break;
case 1: var face = 'west'; break;
case 2: var face = 'up'; break;
case 3: var face = 'down'; break;
default:var face = 'north'; break;
}
return {
event: event,
type: 'cube',
intersects: intersects,
face: face,
cube: obj
}
} else if (intersect.isVertex) {
return {
event: event,
type: 'vertex',
intersects: intersects,
cube: intersect.cube,
vertex: intersect
}
}
} else {
return false;
}
}
render() {
if (this.canvas.isConnected === false) return;
this.controls.update()
this.renderer.render(
display_mode
? display_scene
: scene,
2020-03-05 03:56:17 +08:00
this.camera
2019-07-18 00:02:07 +08:00
)
}
//Camera
2019-12-16 03:04:31 +08:00
get camera() {
return this.isOrtho ? this.camOrtho : this.camPers;
}
2020-03-05 03:56:17 +08:00
setProjectionMode(ortho) {
let position = this.camera.position;
this.isOrtho = !!ortho;
2019-07-18 00:02:07 +08:00
this.resize()
2020-03-05 03:56:17 +08:00
this.controls.object = this.camera;
this.camera.position.copy(position);
if (this.isOrtho) {
this.camera.zoom = 0.5;
this.camOrtho.updateProjectionMatrix()
}
if (Transformer && Transformer.camera !== this.camera) {
Transformer.camera = this.camera;
2019-07-18 00:02:07 +08:00
Transformer.update();
}
2020-03-05 03:56:17 +08:00
this.setLockedAngle()
2019-07-18 00:02:07 +08:00
this.controls.updateSceneScale();
2020-03-05 03:56:17 +08:00
return this;
2019-07-18 00:02:07 +08:00
}
2020-03-05 03:56:17 +08:00
setNormalCamera() {
//Deprecated
this.setProjectionMode(false)
return this;
}
setLockedAngle(angle) {
if (typeof angle === 'number' && this.isOrtho) {
this.angle = angle
this.controls.enableRotate = false;
var dist = 64
switch (angle) {
case 0:
this.camOrtho.axis = 'y'
//this.camOrtho.position.set(0,dist,0)
this.camOrtho.backgroundHandle = [{n: false, a: 'x'}, {n: false, a: 'z'}]
break;
case 1:
this.camOrtho.axis = 'y'
//this.camOrtho.position.set(0,-dist,0)
this.camOrtho.backgroundHandle = [{n: false, a: 'x'}, {n: true, a: 'z'}]
break;
case 2:
this.camOrtho.axis = 'z'
//this.camOrtho.position.set(0,0,dist)
this.camOrtho.backgroundHandle = [{n: false, a: 'x'}, {n: true, a: 'y'}]
break;
case 3:
this.camOrtho.axis = 'z'
//this.camOrtho.position.set(0,0,-dist)
this.camOrtho.backgroundHandle = [{n: true, a: 'x'}, {n: true, a: 'y'}]
break;
case 4:
this.camOrtho.axis = 'x'
//this.camOrtho.position.set(dist,0,0)
this.camOrtho.backgroundHandle = [{n: true, a: 'z'}, {n: true, a: 'y'}]
break;
case 5:
this.camOrtho.axis = 'x'
//this.camOrtho.position.set(-dist,0,0)
this.camOrtho.backgroundHandle = [{n: false, a: 'z'}, {n: true, a: 'y'}]
break;
}
this.loadBackground();
var layer = getAxisNumber(this.camOrtho.axis)+1;
this.camOrtho.layers.set(0);
this.camOrtho.layers.enable(layer);
for (var i = 1; i <= 3; i++) {
if (i != layer) {
this.camOrtho.layers.enable(i+3);
}
2019-07-18 00:02:07 +08:00
}
2020-03-05 03:56:17 +08:00
} else {
this.angle = null;
this.camOrtho.axis = null
this.camOrtho.layers.set(0);
this.camOrtho.layers.enable(4);
this.camOrtho.layers.enable(5);
this.camOrtho.layers.enable(6);
this.resize()
this.controls.enableRotate = true;
this.loadBackground()
2019-07-18 00:02:07 +08:00
}
Transformer.update();
2020-03-05 03:56:17 +08:00
this.loadBackground()
2019-07-18 00:02:07 +08:00
return this;
}
2020-03-05 03:56:17 +08:00
loadAnglePreset(preset) {
if (!preset) return;
this.camera.position.fromArray(preset.position);
this.controls.target.fromArray(preset.target);
if (preset.projection !== 'unset') {
this.setProjectionMode(preset.projection == 'orthographic')
}
if (this.isOrtho && preset.zoom) {
this.camera.zoom = preset.zoom;
this.camera.updateProjectionMatrix()
2019-07-18 00:02:07 +08:00
}
2020-03-05 03:56:17 +08:00
if (!this.isOrtho) {
this.camera.setFocalLength(preset.focal_length||45);
}
this.setLockedAngle(preset.locked_angle)
2019-07-18 00:02:07 +08:00
return this;
}
2020-03-05 03:56:17 +08:00
newAnglePreset() {
let scope = this;
let dialog = new Dialog({
id: 'save_angle',
title: 'menu.preview.save_angle',
width: 540,
form: {
name: {label: 'generic.name'},
projection: {label: 'dialog.save_angle.projection', type: 'select', default: 'unset', options: {
unset: 'generic.unset',
perspective: 'dialog.save_angle.projection.perspective',
orthographic: 'dialog.save_angle.projection.orthographic'
}}
},
onConfirm: function(formResult) {
if (!formResult.name) return;
let preset = {
name: formResult.name,
projection: formResult.projection,
position: scope.camera.position.toArray(),
target: scope.controls.target.toArray(),
}
if (this.isOrtho) preset.zoom = this.camOrtho.zoom;
let presets = localStorage.getItem('camera_presets');
try {
presets = JSON.parse(presets)||[]
} catch (err) {
presets = [];
}
presets.push(preset);
localStorage.setItem('camera_presets', JSON.stringify(presets))
dialog.hide()
}
})
dialog.show()
return this;
}
//Orientation
2019-07-18 00:02:07 +08:00
getFacingDirection() {
var vec = new THREE.Vector3()
this.controls.object.getWorldDirection(vec)
vec.applyAxisAngle(new THREE.Vector3(0, 1, 0), Math.PI / 4).ceil()
switch (vec.x+'_'+vec.z) {
case '1_1':
return 'south'
break;
case '0_0':
return 'north'
break;
case '1_0':
return 'east'
break;
case '0_1':
return 'west'
break;
}
}
getFacingHeight() {
var y = this.controls.object.getWorldDirection(new THREE.Vector3()).y
if (y > 0.5) {
return 'up'
} else if (y < -0.5) {
return 'down';
} else {
return 'middle'
}
}
//Controls
click(event) {
2020-01-24 01:53:36 +08:00
event.preventDefault();
$(':focus').blur();
unselectInterface(event);
2019-12-16 03:04:31 +08:00
convertTouchEvent(event);
this.static_rclick = event.which === 3 || event.type == 'touchstart';
2020-01-24 01:53:36 +08:00
if (event.type == 'touchstart') {
this.rclick_cooldown = setTimeout(() => {
this.rclick_cooldown = true;
}, 420)
}
2019-07-18 00:02:07 +08:00
quad_previews.current = this;
if (Transformer.hoverAxis !== null || (!Keybinds.extra.preview_select.keybind.isTriggered(event) && event.which !== 0)) return;
2019-12-16 03:04:31 +08:00
var data = this.raycast(event);
2019-07-18 00:02:07 +08:00
if (data) {
2019-12-16 03:04:31 +08:00
//this.static_rclick = false
2019-07-18 00:02:07 +08:00
if (Toolbox.selected.selectCubes && Modes.selected.selectCubes && data.type === 'cube') {
if (Toolbox.selected.selectFace) {
main_uv.setFace(data.face, false)
}
2019-08-18 00:26:14 +08:00
Blockbench.dispatchEvent('canvas_select', data)
if (Modes.paint) {
event = 0;
}
2019-08-01 06:01:47 +08:00
if (Format.bone_rig && (
Animator.open ||
(!Format.rotate_cubes && ['rotate_tool', 'pivot_tool'].includes(Toolbox.selected.id)) ||
event.shiftKey
)) {
2019-07-18 00:02:07 +08:00
if (data.cube.parent.type === 'group') {
2019-12-16 03:04:31 +08:00
data.cube.parent.select().showInOutliner();
2019-07-18 00:02:07 +08:00
}
} else {
data.cube.select(event)
}
}
if (typeof Toolbox.selected.onCanvasClick === 'function') {
Toolbox.selected.onCanvasClick(data)
}
return true;
2019-12-16 03:04:31 +08:00
}
if (typeof Toolbox.selected.onCanvasClick === 'function') {
Toolbox.selected.onCanvasClick(0)
}
2020-03-05 03:56:17 +08:00
if (this.angle !== null && this.camOrtho.axis || this.movingBackground) {
2019-07-18 00:02:07 +08:00
this.startSelRect(event)
} else {
return false;
}
}
2019-12-16 03:04:31 +08:00
mousemove(event) {
var data = this.raycast(event);
2019-12-18 00:44:27 +08:00
if (Settings.get('highlight_cubes')) updateCubeHighlights(data && data.cube);
2019-12-16 03:04:31 +08:00
}
2019-07-18 00:02:07 +08:00
raycastMouseCoords(x,y) {
var scope = this;
var canvas_offset = $(scope.canvas).offset()
scope.mouse.x = ((x - canvas_offset.left) / scope.width) * 2 - 1;
scope.mouse.y = - ((y - canvas_offset.top) / scope.height) * 2 + 1;
scope.raycaster.setFromCamera( scope.mouse, scope.camOrtho );
return scope.raycaster.ray.origin
}
2019-12-16 03:04:31 +08:00
occupyTransformer(event) {
2019-07-18 00:02:07 +08:00
Transformer.camera = this.isOrtho ? this.camOrtho : this.camPers
Transformer.orbit_controls = this.controls
Transformer.setCanvas(this.canvas)
main_preview.controls.updateSceneScale()
if (quad_previews) {
quad_previews.hovered = this;
}
2019-12-16 03:04:31 +08:00
if (event && event.type == 'touchstart') {
Transformer.simulateMouseDown(event);
}
2019-07-18 00:02:07 +08:00
return this;
}
2019-12-16 03:04:31 +08:00
showContextMenu(event, force) {
Prop.active_panel = 'preview';
2020-01-24 01:53:36 +08:00
if (this.static_rclick && (event.which === 3 || (event.type == 'touchend' && this.rclick_cooldown == true))) {
2019-12-16 03:04:31 +08:00
var data = this.raycast(event)
2019-07-18 00:02:07 +08:00
if (Toolbox.selected.selectCubes && Modes.selected.selectCubes && data && data.cube) {
2020-03-11 05:19:17 +08:00
if (!Modes.animate) {
data.cube.showContextMenu(event)
}
2019-07-18 00:02:07 +08:00
} else {
this.menu.open(event, this)
}
}
2020-01-24 01:53:36 +08:00
clearTimeout(this.rclick_cooldown);
delete this.rclick_cooldown;
2019-07-18 00:02:07 +08:00
return this;
}
//Selection Rectangle
startSelRect(event) {
var scope = this;
2019-08-18 00:26:14 +08:00
if (Modes.edit || this.movingBackground) {
2019-07-18 00:02:07 +08:00
this.sr_move_f = function(event) { scope.moveSelRect(event)}
this.sr_stop_f = function(event) { scope.stopSelRect(event)}
2019-12-16 03:04:31 +08:00
document.addEventListener('mousemove', this.sr_move_f, false)
2019-07-18 00:02:07 +08:00
document.addEventListener('mouseup', this.sr_stop_f, false)
}
this.selection.start_x = event.offsetX+0
this.selection.start_y = event.offsetY+0
this.selection.client_x = event.clientX+0
2019-12-16 03:04:31 +08:00
this.selection.client_y = event.clientY+0
2019-07-18 00:02:07 +08:00
if (this.movingBackground) {
this.background.before = {
x: this.background.x,
y: this.background.y,
size: this.background.size
}
return
};
2019-08-18 00:26:14 +08:00
if (!Modes.edit) return;
2019-07-18 00:02:07 +08:00
$(this.canvas).parent().append(this.selection.box)
this.selection.activated = settings.canvas_unselect.value;
this.selection.old_selected = selected.slice();
var ray = this.raycastMouseCoords(event.clientX, event.clientY)
this.selection.start_u = ray[this.getUVAxes().u]
this.selection.start_v = ray[this.getUVAxes().v]
this.moveSelRect(event)
}
moveSelRect(event) {
var scope = this;
if (this.movingBackground) {
if (event.shiftKey) {
this.background.size = limitNumber( this.background.before.size + (event.offsetY - this.selection.start_y), 0, 10e3)
} else {
this.background.x = this.background.before.x + (event.offsetX - this.selection.start_x)/this.camOrtho.zoom
this.background.y = this.background.before.y + (event.offsetY - this.selection.start_y)/this.camOrtho.zoom
}
this.updateBackground()
return;
}
var uv_axes = this.getUVAxes()
//Overlay
var c = getRectangle(
2019-12-16 03:04:31 +08:00
Math.clamp(this.selection.start_x, -2, this.width),
Math.clamp(this.selection.start_y, -2, this.height),
Math.clamp(this.selection.start_x + (event.clientX - this.selection.client_x), -2, this.width),
Math.clamp(this.selection.start_y + (event.clientY - this.selection.client_y), -2, this.height),
2019-07-18 00:02:07 +08:00
)
this.selection.box.css('left', c.ax+'px')
this.selection.box.css('top', c.ay+'px')
this.selection.box.css('width', c.x+'px')
this.selection.box.css('height',c.y+'px')
if (c.x + c.y > 40) {
this.selection.activated = true
}
//Select
if (!this.selection.activated) return;
var ray = this.raycastMouseCoords(event.clientX, event.clientY)
var plane_rect = getRectangle(
this.selection.start_u,
this.selection.start_v,
ray[uv_axes.u],
ray[uv_axes.v]
)
unselectAll()
elements.forEach(function(cube) {
2019-08-18 00:26:14 +08:00
if ((event.shiftKey || event.ctrlOrCmd) && scope.selection.old_selected.indexOf(cube) >= 0) {
2019-07-18 00:02:07 +08:00
var isSelected = true
} else {
2019-07-19 23:31:22 +08:00
if (cube instanceof Cube && cube.visibility && cube.mesh) {
2019-07-18 00:02:07 +08:00
var mesh = cube.mesh
var from = new THREE.Vector3().copy(mesh.geometry.vertices[6]).applyMatrix4(mesh.matrixWorld)
var to = new THREE.Vector3().copy(mesh.geometry.vertices[0]).applyMatrix4(mesh.matrixWorld)
var cube_rect = getRectangle(
from[uv_axes.u],
from[uv_axes.v],
to[uv_axes.u],
to[uv_axes.v]
)
var isSelected = doRectanglesOverlap(plane_rect, cube_rect)
} else if (cube instanceof Locator && cube.parent instanceof Group && cube.parent.mesh) {
var mesh = cube.parent.mesh;
var pos = new THREE.Vector3().fromArray(cube.from).applyMatrix4(mesh.matrixWorld);
var cube_rect = getRectangle(
pos[uv_axes.u],
pos[uv_axes.v],
pos[uv_axes.u],
pos[uv_axes.v]
)
var isSelected = doRectanglesOverlap(plane_rect, cube_rect)
}
}
if (isSelected) {
cube.selectLow()
}
})
2020-03-05 03:56:17 +08:00
TickUpdates.selection = true;
2019-07-18 00:02:07 +08:00
}
stopSelRect(event) {
var scope = this;
2019-12-16 03:04:31 +08:00
document.removeEventListener('mousemove', this.sr_move_f)
2019-07-18 00:02:07 +08:00
document.removeEventListener('mouseup', this.sr_stop_f)
if (this.movingBackground) {
delete this.background.before
return
};
this.selection.box.detach()
this.selection.activated = false;
}
getUVAxes() {
switch (this.camOrtho.axis) {
case 'x': return {u: 'z', v: 'y'}; break;
case 'y': return {u: 'x', v: 'z'}; break;
case 'z': return {u: 'x', v: 'y'}; break;
}
}
//Backgrounds
getBackground() {
if (display_mode) {
var id = displayReferenceObjects.active.id
if (id == 'monitor' ||id == 'bow') {
this.background = canvas_scenes.monitor
} else if (['inventory_nine', 'inventory_full', 'hud'].includes(id)) {
this.background = canvas_scenes[id]
} else {
this.background = canvas_scenes.normal
}
2020-03-05 03:56:17 +08:00
} else if (this.angle !== null) {
2019-07-18 00:02:07 +08:00
this.background = canvas_scenes['ortho'+this.angle]
2019-12-16 03:04:31 +08:00
} else {
2019-07-18 00:02:07 +08:00
this.background = canvas_scenes.normal
}
return this.background
}
loadBackground() {
this.getBackground()
if (this.background && this.background.image) {
2019-12-16 03:04:31 +08:00
if (!this.background.imgtag) this.background.imgtag = new Image();
this.background.imgtag.src = this.background.image;
2019-07-18 00:02:07 +08:00
$(this.canvas).css('background-image', 'url("'+this.background.image.split('\\').join('/')+'")')
} else {
$(this.canvas).css('background-image', 'none')
}
this.updateBackground()
return this;
}
updateBackground() {
if (!this.background) return;
var bg = this.background
2020-03-05 03:56:17 +08:00
var zoom = (this.angle !== null && bg.lock === true) ? this.camOrtho.zoom : 1
2019-07-18 00:02:07 +08:00
var pos_x = 0;
var pos_y = 0;
2020-03-05 03:56:17 +08:00
if (this.angle !== null && bg.lock !== false) {
2019-07-18 00:02:07 +08:00
pos_x = this.camOrtho.backgroundHandle[0].n === true ? 1 : -1
pos_x *= this.controls.target[this.camOrtho.backgroundHandle[0].a] * zoom * 40
pos_y = this.camOrtho.backgroundHandle[1].n === true ? 1 : -1
pos_y *= this.controls.target[this.camOrtho.backgroundHandle[1].a] * zoom * 40
}
pos_x += (bg.x * zoom) + this.width/2 - ( bg.size * zoom) / 2
pos_y += (bg.y * zoom) + this.height/2 -((bg.size / bg.ratio||1) * zoom) / 2
$(this.canvas).css('background-position-x', pos_x + 'px')
$(this.canvas).css('background-position-y', pos_y + 'px')
$(this.canvas).css('background-size', bg.size * zoom +'px')
return this;
}
clearBackground() {
this.loadBackground()
this.background.image = false
this.background.size = limitNumber(this.background.size, 100, 2400)
this.background.x = limitNumber(this.background.x, 0, this.width-30)
this.background.y = limitNumber(this.background.y, 0, this.height-30)
this.loadBackground()
return this;
}
startMovingBackground() {
2019-12-16 03:04:31 +08:00
if (this.movingBackground) {
this.stopMovingBackground()
}
2019-07-18 00:02:07 +08:00
this.movingBackground = true;
this.controls.enabled_before = this.controls.enabled
this.controls.enabled = false
Blockbench.showMessageBox({
translateKey: 'drag_background',
icon: 'open_with'
})
}
stopMovingBackground() {
this.movingBackground = false;
this.controls.enabled = this.controls.enabled_before
delete this.controls.enabled_before
}
backgroundPositionDialog() {
var scope = this;
2019-12-16 03:04:31 +08:00
if (this.movingBackground) {
this.stopMovingBackground()
}
2019-07-18 00:02:07 +08:00
var dialog = new Dialog({
id: 'background_position',
title: tl('message.set_background_position.title'),
lines: [
`<div class="dialog_bar">
<input type="number" class="dark_bordered" value="${scope.background.x}" id="background_pos_x">
<input type="number" class="dark_bordered" value="${scope.background.y}" id="background_pos_y">
<input type="number" class="dark_bordered" value="${scope.background.size}" id="background_size">
</div>`
],
onConfirm: function() {
var coords = [
parseFloat( $(dialog.object).find('#background_pos_x').val() ),
parseFloat( $(dialog.object).find('#background_pos_y').val() ),
parseFloat( $(dialog.object).find('#background_size').val() )
]
dialog.hide()
if (!scope.background) return;
if (!isNaN(coords[0])) { scope.background.x = coords[0] }
if (!isNaN(coords[1])) { scope.background.y = coords[1] }
if (!isNaN(coords[2])) { scope.background.size = coords[2] }
scope.updateBackground()
}
})
dialog.show()
}
//Misc
screenshot(options, cb) {
var scope = this;
2019-12-16 03:04:31 +08:00
if (!options) options = 0;
2019-07-18 00:02:07 +08:00
2019-09-04 15:52:21 +08:00
Canvas.withoutGizmos(function() {
2019-07-18 00:02:07 +08:00
2019-09-04 15:52:21 +08:00
scope.render()
var dataUrl = scope.canvas.toDataURL()
2019-07-18 00:02:07 +08:00
2019-12-23 00:33:55 +08:00
if (options.crop == false && !options.width && !options.height) {
Screencam.returnScreenshot(dataUrl, cb)
}
2019-09-04 15:52:21 +08:00
dataUrl = dataUrl.replace('data:image/png;base64,','')
Jimp.read(Buffer.from(dataUrl, 'base64')).then(function(image) {
2019-07-18 00:02:07 +08:00
2019-12-23 00:33:55 +08:00
if (display_mode && display_slot === 'gui' && options.crop !== false) {
2019-09-04 15:52:21 +08:00
var zoom = display_preview.camOrtho.zoom * devicePixelRatio
var resolution = 256 * zoom;
var start_x = display_preview.width *devicePixelRatio/2 - display_preview.controls.target.x*zoom*40 - resolution/2;
var start_y = display_preview.height*devicePixelRatio/2 + display_preview.controls.target.y*zoom*40 - resolution/2;
image.crop(start_x, start_y, resolution, resolution)
} else {
2019-12-16 03:04:31 +08:00
if (options.crop !== false) {
image.autocrop([0, false])
}
2019-09-04 15:52:21 +08:00
if (options && options.width && options.height) {
image.contain(options.width, options.height)
}
2019-07-18 00:02:07 +08:00
}
2019-09-04 15:52:21 +08:00
image.getBase64(Jimp.MIME_PNG, function(a, dataUrl){
Screencam.returnScreenshot(dataUrl, cb)
})
});
2019-07-18 00:02:07 +08:00
})
2019-09-04 15:52:21 +08:00
2019-07-18 00:02:07 +08:00
}
fullscreen() {
if (quad_previews.current) {
quad_previews.current.controls.stopMovement()
}
quad_previews.current = this;
quad_previews.enabled = false;
$('#preview').empty()
var wrapper = $('<div class="single_canvas_wrapper"></div>')
wrapper.append(this.canvas)
$('#preview').append(wrapper)
previews.forEach(function(prev) {
if (prev.canvas.isConnected) {
prev.resize()
}
})
if (Interface.data) {
updateInterfacePanels()
}
return this;
}
toggleFullscreen() {
if (quad_previews.enabled) {
this.fullscreen()
} else {
openQuadView()
}
}
}
Preview.prototype.menu = new Menu([
2020-03-05 03:56:17 +08:00
'screenshot_model',
2019-07-18 00:02:07 +08:00
{icon: 'icon-player', name: 'settings.display_skin', condition: () => (display_mode && displayReferenceObjects.active.id === 'player'), click: function() {
changeDisplaySkin()
}},
2020-03-05 03:56:17 +08:00
'preview_checkerboard',
{icon: 'wallpaper', name: 'menu.preview.background', children(preview) {
2019-07-18 00:02:07 +08:00
var has_background = !!preview.background.image
return [
{icon: 'folder', name: 'menu.preview.background.load', click: function(preview) {
Blockbench.import({
extensions: ['png', 'jpg', 'jpeg', 'bmp', 'tiff', 'tif', 'gif'],
type: 'Image',
readtype: 'image'
}, function(files) {
if (files) {
preview.background.image = isApp ? files[0].path : files[0].content
preview.loadBackground()
}
}, 'image', false)
}},
2020-01-24 01:53:36 +08:00
{icon: 'fa-clipboard', name: 'menu.preview.background.clipboard', condition: isApp, click: function(preview) {
var image = clipboard.readImage().toDataURL();
if (image.length > 32) {
preview.background.image = image;
preview.loadBackground();
}
}},
2019-07-18 00:02:07 +08:00
{icon: 'photo_size_select_large', name: 'menu.preview.background.position', condition: has_background, click: function(preview) {
preview.startMovingBackground()
}},
{icon: 'photo_size_select_large', name: 'menu.preview.background.set_position', condition: has_background, click: function(preview) {
preview.backgroundPositionDialog()
}},
{
name: 'menu.preview.background.lock',
2020-03-05 03:56:17 +08:00
condition: (has_background && preview.background.lock !== null && preview.angle !== null),
2019-07-18 00:02:07 +08:00
icon: preview.background.lock?'check_box':'check_box_outline_blank',
click: function(preview) {
preview.background.lock = !preview.background.lock
preview.updateBackground()
}},
2020-03-05 03:56:17 +08:00
{icon: 'clear', name: 'generic.remove', condition: has_background, click: function(preview) {
2019-07-18 00:02:07 +08:00
preview.clearBackground()
}}
]
}},
2020-03-05 03:56:17 +08:00
'_',
{icon: 'add_a_photo', name: 'menu.preview.save_angle', condition(preview) {return !preview.movingBackground && !Modes.display}, click(preview) {
preview.newAnglePreset()
}},
{icon: 'videocam', name: 'menu.preview.angle', condition(preview) {return !preview.movingBackground && !Modes.display}, children: function(preview) {
var children = [
2019-07-18 00:02:07 +08:00
]
2020-03-05 03:56:17 +08:00
let presets = localStorage.getItem('camera_presets')
presets = (presets && JSON.parse(presets)) || [];
let all_presets = [...DefaultCameraPresets, ...presets];
all_presets.forEach(preset => {
let icon = typeof preset.locked_angle !== 'number' ? 'videocam' : (preset.locked_angle == preview.angle ? 'radio_button_checked' : 'radio_button_unchecked');
children.push({
name: preset.name,
color: preset.color,
id: preset.name,
icon,
click: preset.default ? () => {
preview.loadAnglePreset(preset)
} : null,
children: !preset.default && [
{icon: 'check_circle', name: 'menu.preview.angle.load', click() {
preview.loadAnglePreset(preset)
}},
{icon: 'delete', name: 'generic.delete', click() {
presets.remove(preset)
localStorage.setItem('camera_presets', JSON.stringify(presets))
}}
]
})
})
return children;
2019-07-18 00:02:07 +08:00
}},
2020-03-05 03:56:17 +08:00
{icon: (preview) => (preview.isOrtho ? 'check_box' : 'check_box_outline_blank'), name: 'menu.preview.orthographic', click: function(preview) {
preview.setProjectionMode(!preview.isOrtho);
}},
'_',
2019-08-18 00:26:14 +08:00
{icon: 'widgets', name: 'menu.preview.quadview', condition: function(preview) {return !quad_previews.enabled && !preview.movingBackground && !Modes.display && !Animator.open}, click: function() {
2019-07-18 00:02:07 +08:00
openQuadView()
}},
2020-03-05 03:56:17 +08:00
{icon: 'web_asset', name: 'menu.preview.maximize', condition: function(preview) {return quad_previews.enabled && !preview.movingBackground && !Modes.display}, click: function(preview) {
2019-07-18 00:02:07 +08:00
preview.fullscreen()
}},
{icon: 'cancel', color: 'x', name: 'menu.preview.stop_drag', condition: function(preview) {return preview.movingBackground;}, click: function(preview) {
preview.stopMovingBackground()
}}
])
function openQuadView() {
quad_previews.enabled = true;
$('#preview').empty()
var wrapper1 = $('<div class="quad_canvas_wrapper qcw_x qcw_y"></div>')
wrapper1.append(quad_previews.one.canvas)
$('#preview').append(wrapper1)
var wrapper2 = $('<div class="quad_canvas_wrapper qcw_y"></div>')
wrapper2.append(quad_previews.two.canvas)
$('#preview').append(wrapper2)
var wrapper3 = $('<div class="quad_canvas_wrapper qcw_x"></div>')
wrapper3.append(quad_previews.three.canvas)
$('#preview').append(wrapper3)
var wrapper4 = $('<div class="quad_canvas_wrapper"></div>')
wrapper4.append(quad_previews.four.canvas)
$('#preview').append(wrapper4)
updateInterface()
}
2019-08-01 06:01:47 +08:00
const Screencam = {
2019-12-16 03:04:31 +08:00
recording_timelapse: false,
2019-08-01 06:01:47 +08:00
fullScreen(options, cb) {
setTimeout(function() {
2020-03-05 03:56:17 +08:00
currentwindow.capturePage().then(function(screenshot) {
2019-08-01 06:01:47 +08:00
var dataUrl = screenshot.toDataURL()
2020-03-05 03:56:17 +08:00
if (!(options && options.width && options.height)) {
Screencam.returnScreenshot(dataUrl, cb)
return;
}
2019-08-01 06:01:47 +08:00
dataUrl = dataUrl.replace('data:image/png;base64,','')
Jimp.read(Buffer.from(dataUrl, 'base64')).then(function(image) {
2020-03-05 03:56:17 +08:00
image.contain(options.width, options.height)
2019-08-01 06:01:47 +08:00
image.getBase64(Jimp.MIME_PNG, function(a, dataUrl){
Screencam.returnScreenshot(dataUrl, cb)
})
});
})
}, 40)
},
returnScreenshot(dataUrl, cb) {
if (cb) {
cb(dataUrl)
} else if (isApp) {
var screenshot = nativeImage.createFromDataURL(dataUrl)
var img = new Image()
var is_gif = dataUrl.substr(5, 9) == 'image/gif'
img.src = dataUrl
var btns = [tl('dialog.cancel'), tl('dialog.save')]
if (!is_gif) {
btns.push(tl('message.screenshot.clipboard'))
}
Blockbench.showMessageBox({
translateKey: 'screenshot',
icon: img,
buttons: btns,
confirm: 1,
cancel: 0
}, function(result) {
if (result === 1) {
2019-09-06 06:16:54 +08:00
Blockbench.export()
2019-09-04 15:37:38 +08:00
ElecDialogs.showSaveDialog(currentwindow, {filters: [ {name: tl('data.image'), extensions: [is_gif ? 'gif' : 'png']} ]}, function (fileName) {
2019-08-01 06:01:47 +08:00
if (fileName === undefined) {
return;
}
fs.writeFile(fileName, Buffer(dataUrl.split(',')[1], 'base64'), err => {})
})
} else if (result === 2) {
clipboard.writeImage(screenshot)
}
})
} else {
new Dialog({
title: tl('message.screenshot.right_click'),
id: 'screenie',
lines: ['<img src="'+dataUrl+'" width="600px" class="allow_default_menu"></img>'],
draggable: true,
singleButton: true
}).show()
}
},
cleanCanvas(options, cb) {
quad_previews.current.screenshot(options, cb)
},
createGif(options, cb) {
2020-03-05 03:56:17 +08:00
if (typeof options !== 'object') options = {}
if (!options.length_mode) options.length_mode = 'seconds';
if (!options.length) options.length = 1;
2019-08-01 06:01:47 +08:00
var preview = quad_previews.current;
2020-03-05 03:56:17 +08:00
var animation = Animator.selected;
2019-08-01 06:01:47 +08:00
var interval = options.fps ? (1000/options.fps) : 100;
2020-03-05 03:56:17 +08:00
var frames = 0;
const gif = new GIF({
2019-08-01 06:01:47 +08:00
repeat: options.repeat,
quality: options.quality,
2020-03-05 03:56:17 +08:00
background: {r: 30, g: 0, b: 255},
transparent: 0x1e01ff,
2019-08-01 06:01:47 +08:00
});
if (options.turnspeed) {
preview.controls.autoRotate = true;
preview.controls.autoRotateSpeed = options.turnspeed;
2020-03-05 03:56:17 +08:00
preview.controls.autoRotateProgress = 0;
} else if (options.length_mode == 'turntable') {
options.length_mode = 'seconds'
}
if (options.play && animation) {
Timeline.time = 0;
Timeline.start()
if (!animation.length) options.length_mode = 'seconds';
} else if (options.length_mode == 'animation') {
options.length_mode = 'seconds'
2019-08-01 06:01:47 +08:00
}
if (!options.silent) {
Blockbench.setStatusBarText(tl('status_bar.recording_gif'));
gif.on('progress', Blockbench.setProgress);
}
2020-03-05 03:56:17 +08:00
function getProgress() {
switch (options.length_mode) {
case 'seconds': return interval*frames/(options.length*1000); break;
case 'frames': return frames/options.length; break;
case 'turntable': return Math.abs(preview.controls.autoRotateProgress) / (2*Math.PI); break;
case 'animation': return Timeline.time / (animation.length-(interval/1000)); break;
}
}
2019-08-01 06:01:47 +08:00
var loop = setInterval(() => {
frames++;
2020-03-05 03:56:17 +08:00
if (getProgress() >= 1) {
endRecording()
return;
}
2019-09-04 15:52:21 +08:00
Canvas.withoutGizmos(function() {
var img = new Image();
preview.render();
img.src = preview.canvas.toDataURL();
img.onload = () => {
gif.addFrame(img, {delay: interval});
2019-08-01 06:01:47 +08:00
}
2019-09-04 15:52:21 +08:00
})
2020-03-05 03:56:17 +08:00
Blockbench.setProgress(getProgress());
2019-08-01 06:01:47 +08:00
}, interval)
2019-12-16 03:04:31 +08:00
2020-03-05 03:56:17 +08:00
function endRecording() {
2019-12-16 03:04:31 +08:00
gif.render();
clearInterval(loop)
if (!options.silent) {
Blockbench.setStatusBarText(tl('status_bar.processing_gif'))
}
if (Animator.open && Timeline.playing) {
Timeline.pause();
}
if (options.turnspeed) {
preview.controls.autoRotate = false;
}
2020-03-05 03:56:17 +08:00
}
2019-12-16 03:04:31 +08:00
gif.on('finished', blob => {
var reader = new FileReader();
reader.onload = () => {
if (!options.silent) {
Blockbench.setProgress(0);
Blockbench.setStatusBarText();
}
Screencam.returnScreenshot(reader.result, cb);
}
reader.readAsDataURL(blob);
});
},
recordTimelapse(options) {
if (!options.destination) return;
2019-12-18 00:44:27 +08:00
function getFileName(num) {
return `${Project.name||'model'}_${num.toDigitString(4)}.png`;
}
var index = 0;
try {
var list = fs.readdirSync(options.destination);
while (list.includes(getFileName(index+1))) {
index++;
}
} catch (err) {
console.log('Unable to analyze past timelapse recording', err)
}
2019-12-16 03:04:31 +08:00
Prop.recording = true;
BarItems.timelapse.setIcon('pause');
Blockbench.showQuickMessage('message.timelapse_start');
function saveImage(image) {
2019-12-18 00:44:27 +08:00
var path = `${options.destination}${osfs}${getFileName(index)}`;
2019-12-16 03:04:31 +08:00
fs.writeFile(path, image, (e, b) => {});
}
if (options.source === 'locked') {
var view_pos = new THREE.Vector3().copy(quad_previews.current.camera.position);
var view_tar = new THREE.Vector3().copy(quad_previews.current.controls.target);
}
Screencam.timelapse_loop = setInterval(function() {
index++;
if (!isApp || options.source === 'preview' || options.source === 'locked') {
var scope = quad_previews.current;
if (options.source === 'locked') {
var old_pos = new THREE.Vector3().copy(scope.camera.position);
var old_tar = new THREE.Vector3().copy(scope.controls.target);
scope.camera.position.copy(view_pos);
scope.controls.target.copy(view_tar);
}
Canvas.withoutGizmos(function() {
scope.render();
var dataUrl = scope.canvas.toDataURL();
saveImage(nativeImage.createFromDataURL(dataUrl).toPNG());
if (options.source === 'locked') {
scope.camera.position.copy(old_pos);
scope.controls.target.copy(old_tar);
}
})
} else {
currentwindow.capturePage((image) => {
saveImage(image.toPNG());
});
}
}, options.interval*1000);
},
stopTimelapse() {
if (Prop.recording) {
Prop.recording = false;
clearInterval(Screencam.timelapse_loop);
BarItems.timelapse.setIcon('timelapse');
Blockbench.showQuickMessage('message.timelapse_stop');
}
2019-08-01 06:01:47 +08:00
}
}
2019-07-18 00:02:07 +08:00
//Init/Update
function initCanvas() {
previews = []
//Objects
scene = new THREE.Scene();
display_scene = new THREE.Scene();
display_area = new THREE.Object3D();
display_base = new THREE.Object3D();
display_scene.add(display_area)
display_area.add(display_base)
scene.name = 'scene'
display_base.name = 'display_base'
display_area.name = 'display_area'
display_scene.name = 'display_scene'
scene.add(Vertexsnap.vertexes)
Vertexsnap.vertexes.name = 'vertex_handles'
outlines = new THREE.Object3D();
outlines.name = 'outline_group'
scene.add(outlines)
var DScene = function(data) {
data = data||{}
this.name = data.name ? tl(data.name) : ''
this.image = data.image||false
this.size = data.size||1000
this.x = data.x||0
this.y = data.y||0
this.lock = data.lock||false
}
canvas_scenes = {
normal: new DScene({name: 'menu.preview.perspective.normal', lock: null}),
ortho0: new DScene({name: 'direction.top', lock: true}),
ortho1: new DScene({name: 'direction.bottom', lock: true}),
ortho2: new DScene({name: 'direction.south', lock: true}),
ortho3: new DScene({name: 'direction.north', lock: true}),
ortho4: new DScene({name: 'direction.east', lock: true}),
ortho5: new DScene({name: 'direction.west', lock: true}),
monitor: new DScene({name: 'display.reference.monitor' }),
inventory_nine: new DScene({name: 'display.reference.inventory_nine', image: './assets/inventory_nine.png', x: 0, y: -525, size: 1051, lock: true}),
inventory_full: new DScene({name: 'display.reference.inventory_full', image: './assets/inventory_full.png', x: 0, y: -1740, size: 2781, lock: true}),
hud: new DScene({name: 'display.reference.hud', image: './assets/hud.png', x: -224, y: -447.5, size: 3391, lock: true}),
}
if (localStorage.getItem('canvas_scenes')) {
var stored_canvas_scenes = undefined;
try {
stored_canvas_scenes = JSON.parse(localStorage.getItem('canvas_scenes'))
} catch (err) {}
if (stored_canvas_scenes) {
for (var key in canvas_scenes) {
if (stored_canvas_scenes.hasOwnProperty(key)) {
let store = stored_canvas_scenes[key]
let real = canvas_scenes[key]
if (store.image !== undefined) {real.image = store.image}
if (store.size !== undefined) {real.size = store.size}
if (store.x !== undefined) {real.x = store.x}
if (store.y !== undefined) {real.y = store.y}
if (store.lock !== undefined) {real.lock = store.lock}
}
}
}
}
active_scene = canvas_scenes.normal
main_preview = new Preview({id: 'main'}).fullscreen()
//TransformControls
Transformer = new THREE.TransformControls(main_preview.camPers, main_preview.canvas)
Transformer.setSize(0.5)
scene.add(Transformer)
main_preview.occupyTransformer()
//Light
Sun = new THREE.AmbientLight( 0xffffff );
Sun.name = 'sun'
scene.add(Sun);
Sun.intensity = 0.44
lights = new THREE.Object3D()
lights.name = 'lights'
var light_top = new THREE.DirectionalLight();
light_top.name = 'light_top'
light_top.position.set(8, 100, 8)
lights.add(light_top);
light_top.intensity = 0.66
var light_north = new THREE.DirectionalLight();
light_north.name = 'light_north'
light_north.position.set(8, 8, -100)
lights.add(light_north);
var light_south = new THREE.DirectionalLight();
light_south.name = 'light_south'
light_south.position.set(8, 8, 100)
lights.add(light_south);
light_north.intensity = light_south.intensity = 0.44
var light_west = new THREE.DirectionalLight();
light_west.name = 'light_west'
light_west.position.set(-100, 8, 8)
lights.add(light_west);
var light_east = new THREE.DirectionalLight();
light_east.name = 'light_east'
light_east.position.set(100, 8, 8)
lights.add(light_east);
light_west.intensity = light_east.intensity = 0.22
setShading()
quad_previews = {
2020-03-05 03:56:17 +08:00
one: new Preview({id: 'one'}).loadAnglePreset(DefaultCameraPresets[1]),
2019-07-18 00:02:07 +08:00
two: main_preview,
2020-03-05 03:56:17 +08:00
three: new Preview({id: 'three'}).loadAnglePreset(DefaultCameraPresets[3]),
four: new Preview({id: 'four'}).loadAnglePreset(DefaultCameraPresets[5]),
2019-07-18 00:02:07 +08:00
current: main_preview
}
//emptyMaterial
var img = new Image()
img.src = 'assets/missing.png'
var tex = new THREE.Texture(img)
img.tex = tex;
img.tex.magFilter = THREE.NearestFilter
img.tex.minFilter = THREE.NearestFilter
img.onload = function() {
this.tex.needsUpdate = true;
}
emptyMaterials = []
2020-01-24 01:53:36 +08:00
markerColors.forEach(function(s, i) {
2019-07-18 00:02:07 +08:00
var thismaterial = new THREE.MeshLambertMaterial({
color: 0xffffff,
2019-12-16 03:04:31 +08:00
vertexColors: THREE.FaceColors,
2019-07-18 00:02:07 +08:00
map: tex
})
2020-01-24 01:53:36 +08:00
thismaterial.color.set(s.pastel)
2019-07-18 00:02:07 +08:00
emptyMaterials.push(thismaterial)
})
var img = new Image();
img.src = 'assets/north.png';
var tex = new THREE.Texture(img);
img.tex = tex;
img.tex.magFilter = THREE.NearestFilter;
img.tex.minFilter = THREE.NearestFilter;
img.onload = function() {
this.tex.needsUpdate = true;
}
2019-12-16 03:04:31 +08:00
Canvas.northMarkMaterial = new THREE.MeshBasicMaterial({
2019-07-18 00:02:07 +08:00
map: tex,
transparent: true,
side: THREE.DoubleSide,
alphaTest: 0.2
})
//Rotation Pivot
var helper1 = new THREE.AxesHelper(2)
var helper2 = new THREE.AxesHelper(2)
helper1.rotation.x = Math.PI / 1
helper2.rotation.x = Math.PI / -1
helper2.rotation.y = Math.PI / 1
helper2.scale.y = -1
rot_origin.add(helper1)
rot_origin.add(helper2)
rot_origin.rotation.reorder('ZYX')
2020-01-24 01:53:36 +08:00
rot_origin.base_scale = new THREE.Vector3(1, 1, 1);
2020-03-05 03:56:17 +08:00
rot_origin.no_export = true;
2019-07-18 00:02:07 +08:00
setupGrid = true;
resizeWindow()
}
function animate() {
TickUpdates.Run()
requestAnimationFrame( animate );
previews.forEach(function(prev) {
prev.render()
})
framespersecond++;
if (display_mode === true && ground_animation === true && !Transformer.hoverAxis) {
DisplayMode.groundAnimation()
}
}
function setShading() {
scene.remove(lights)
display_scene.remove(lights)
Sun.intensity = settings.brightness.value/100;
if (settings.shading.value === true) {
(display_mode ? display_scene : scene).add(lights)
} else {
Sun.intensity *= (1/0.6)
}
}
2019-12-18 00:44:27 +08:00
function updateCubeHighlights(hover_cube, force_off) {
Cube.all.forEach(cube => {
if (cube.visibility) {
var mesh = cube.mesh;
mesh.geometry.faces.forEach(face => {
var b_before = face.color.b;
if (Settings.get('highlight_cubes') && (hover_cube == cube || cube.selected) && Modes.edit && !force_off) {
face.color.setRGB(1.3, 1.32, 1.34);
} else {
face.color.setRGB(1, 1, 1);
}
if (face.color.b != b_before) {
mesh.geometry.colorsNeedUpdate = true;
}
})
}
})
}
2019-07-18 00:02:07 +08:00
//Helpers
function buildGrid() {
three_grid.children.length = 0;
if (Canvas.side_grids) {
Canvas.side_grids.x.children.length = 0;
Canvas.side_grids.z.children.length = 0;
}
if (Modes.display && settings.display_grid.value === false) return;
three_grid.name = 'grid_group'
2019-12-16 03:04:31 +08:00
gizmo_colors.grid.set(parseInt('0x'+CustomTheme.data.colors.grid.replace('#', ''), 16));
2019-07-18 00:02:07 +08:00
var material;
2019-12-16 03:04:31 +08:00
Canvas.northMarkMaterial.color = gizmo_colors.grid
2019-07-18 00:02:07 +08:00
function setupAxisLine(origin, length, axis) {
var color = 'rgb'[getAxisNumber(axis)]
var geometry = new THREE.Geometry();
var material = new THREE.LineBasicMaterial({color: gizmo_colors[color]});
var dest = new THREE.Vector3().copy(origin)
dest[axis] += length
geometry.vertices.push(origin)
geometry.vertices.push(dest)
var line = new THREE.Line( geometry, material);
line.name = 'axis_line_'+axis;
three_grid.add(line)
}
//Axis Lines
if (settings.base_grid.value || settings.full_grid.value)
2019-12-16 03:04:31 +08:00
if (Format.centered_grid || !settings.full_grid.value) {
var length = Format.centered_grid
2019-07-18 00:02:07 +08:00
? (settings.full_grid.value ? 24 : 8)
: 16
setupAxisLine(new THREE.Vector3( 0, 0.001, 0), length, 'x')
setupAxisLine(new THREE.Vector3( 0, 0.001, 0), length, 'z')
} else {
setupAxisLine(new THREE.Vector3( -16, 0.001, -16), 48, 'x')
setupAxisLine(new THREE.Vector3( -16, 0.001, -16), 48, 'z')
}
var side_grid = new THREE.Object3D()
if (settings.full_grid.value === true) {
//Grid
2019-12-16 03:04:31 +08:00
var grid = new THREE.GridHelper(48, 48/canvasGridSize(), gizmo_colors.grid)
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
grid.position.set(0,0,0)
} else {
grid.position.set(8,0,8)
}
grid.name = 'grid'
three_grid.add(grid)
side_grid.add(grid.clone())
//North
geometry = new THREE.PlaneGeometry(5, 5)
2019-12-16 03:04:31 +08:00
var north_mark = new THREE.Mesh(geometry, Canvas.northMarkMaterial)
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
north_mark.position.set(0,0,-27)
} else {
north_mark.position.set(8,0,-19)
}
north_mark.rotation.x = Math.PI / -2
three_grid.add(north_mark)
} else {
if (settings.large_grid.value === true) {
//Grid
2019-12-16 03:04:31 +08:00
var grid = new THREE.GridHelper(48, 3, gizmo_colors.grid)
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
grid.position.set(0,0,0)
} else {
grid.position.set(8,0,8)
}
grid.name = 'grid'
three_grid.add(grid)
side_grid.add(grid.clone())
}
if (settings.base_grid.value === true) {
//Grid
2019-12-16 03:04:31 +08:00
var grid = new THREE.GridHelper(16, 16/canvasGridSize(), gizmo_colors.grid)
2019-07-18 00:02:07 +08:00
2019-12-16 03:04:31 +08:00
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
grid.position.set(0,0,0)
} else {
grid.position.set(8,0,8)
}
grid.name = 'grid'
three_grid.add(grid)
side_grid.add(grid.clone())
//North
geometry = new THREE.PlaneGeometry(2.4, 2.4)
2019-12-16 03:04:31 +08:00
var north_mark = new THREE.Mesh(geometry, Canvas.northMarkMaterial)
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
north_mark.position.set(0,0,-9.5)
} else {
north_mark.position.set(8,0,-1.5)
}
north_mark.rotation.x = Math.PI / -2
three_grid.add(north_mark)
}
}
if (settings.large_box.value === true) {
var geometry_box = new THREE.EdgesGeometry(new THREE.BoxBufferGeometry(48, 48, 48));
var line_material = new THREE.LineBasicMaterial({color: gizmo_colors.grid});
var large_box = new THREE.LineSegments( geometry_box, line_material);
2019-12-16 03:04:31 +08:00
if (Format.centered_grid) {
2019-07-18 00:02:07 +08:00
large_box.position.set(0,8,0)
} else {
large_box.position.set(8,8,8)
}
large_box.name = 'grid'
three_grid.add(large_box)
}
scene.add(three_grid)
Canvas.side_grids = {
x: side_grid,
z: side_grid.clone()
}
2020-03-11 05:19:17 +08:00
three_grid.add(Canvas.side_grids.x)
2019-07-18 00:02:07 +08:00
Canvas.side_grids.x.name = 'side_grid_x'
Canvas.side_grids.x.visible = !Modes.display;
Canvas.side_grids.x.rotation.z = Math.PI/2;
2019-12-16 03:04:31 +08:00
Canvas.side_grids.x.position.y = Format.centered_grid ? 8 : 0;
2020-03-15 04:32:24 +08:00
Canvas.side_grids.z.position.y = -512
2019-07-18 00:02:07 +08:00
Canvas.side_grids.x.children.forEach(el => {
el.layers.set(1)
});
2020-03-11 05:19:17 +08:00
three_grid.add(Canvas.side_grids.z)
2020-03-05 03:56:17 +08:00
Canvas.side_grids.z.name = 'side_grid_z'
2019-07-18 00:02:07 +08:00
Canvas.side_grids.z.visible = !Modes.display;
Canvas.side_grids.z.rotation.z = Math.PI/2;
2020-03-05 03:56:17 +08:00
Canvas.side_grids.z.rotation.y = Math.PI/2
2019-12-16 03:04:31 +08:00
Canvas.side_grids.z.position.y = Format.centered_grid ? 8 : 0;
2020-03-15 04:32:24 +08:00
Canvas.side_grids.z.position.z = -512
2019-07-18 00:02:07 +08:00
Canvas.side_grids.z.children.forEach(el => {
el.layers.set(3)
});
}
BARS.defineActions(function() {
2019-08-18 00:26:14 +08:00
new Action('toggle_wireframe', {
2019-07-18 00:02:07 +08:00
icon: 'border_clear',
category: 'view',
keybind: new Keybind({key: 90}),
condition: () => Toolbox && Toolbox.selected && Toolbox.selected.allowWireframe,
click: function () {
Prop.wireframe = !Prop.wireframe
2019-09-06 06:16:54 +08:00
Canvas.updateAllFaces()
2019-07-18 00:02:07 +08:00
if (Modes.id === 'animate') {
Animator.preview()
}
Blockbench.showQuickMessage('message.wireframe.' + (Prop.wireframe ? 'enabled' : 'disabled'))
}
})
2020-03-05 03:56:17 +08:00
new Action('preview_checkerboard', {
name: tl('settings.preview_checkerboard'),
description: tl('settings.preview_checkerboard.desc'),
2019-12-16 03:04:31 +08:00
category: 'view',
2020-03-05 03:56:17 +08:00
linked_setting: 'preview_checkerboard',
2019-12-16 03:04:31 +08:00
keybind: new Keybind({key: 84}),
click: function () {
2020-03-05 03:56:17 +08:00
this.toggleLinkedSetting()
}
})
new Action('uv_checkerboard', {
name: tl('settings.uv_checkerboard'),
description: tl('settings.uv_checkerboard.desc'),
category: 'view',
linked_setting: 'uv_checkerboard',
click: function () {
this.toggleLinkedSetting()
}
})
new Action('toggle_shading', {
name: tl('settings.shading'),
description: tl('settings.shading.desc'),
category: 'view',
linked_setting: 'shading',
click: function () {
this.toggleLinkedSetting()
2019-12-16 03:04:31 +08:00
}
})
2019-07-18 00:02:07 +08:00
2019-08-18 00:26:14 +08:00
new Action('screenshot_model', {
2019-07-18 00:02:07 +08:00
icon: 'fa-cubes',
category: 'view',
keybind: new Keybind({key: 80, ctrl: true}),
click: function () {quad_previews.current.screenshot()}
})
2019-08-18 00:26:14 +08:00
new Action('record_model_gif', {
2019-07-18 00:02:07 +08:00
icon: 'local_movies',
category: 'view',
click: function () {
new Dialog({
id: 'create_gif',
title: tl('dialog.create_gif.title'),
draggable: true,
form: {
2020-03-05 03:56:17 +08:00
length_mode: {label: 'dialog.create_gif.length_mode', type: 'select', default: 'seconds', options: {
seconds: 'dialog.create_gif.length_mode.seconds',
frames: 'dialog.create_gif.length_mode.frames',
animation: 'dialog.create_gif.length_mode.animation',
turntable: 'dialog.create_gif.length_mode.turntable',
}},
2019-07-18 00:02:07 +08:00
length: {label: 'dialog.create_gif.length', type: 'number', value: 10, step: 0.25},
fps: {label: 'dialog.create_gif.fps', type: 'number', value: 10},
2019-08-01 06:01:47 +08:00
quality:{label: 'dialog.create_gif.compression', type: 'number', value: 20, min: 1, max: 80},
2019-07-18 00:02:07 +08:00
turn: {label: 'dialog.create_gif.turn', type: 'number', value: 0, min: -10, max: 10},
play: {label: 'dialog.create_gif.play', type: 'checkbox', condition: Animator.open},
},
onConfirm: function(formData) {
Screencam.createGif({
2020-03-05 03:56:17 +08:00
length_mode: formData.length_mode,
length: limitNumber(formData.length, 0.1, 24000),
2019-07-18 00:02:07 +08:00
fps: limitNumber(formData.fps, 0.5, 30),
quality: limitNumber(formData.quality, 0, 30),
2020-03-05 03:56:17 +08:00
play: formData.play,
2019-07-18 00:02:07 +08:00
turnspeed: formData.turn,
}, Screencam.returnScreenshot)
this.hide()
}
}).show()
}
})
2019-12-16 03:04:31 +08:00
new Action('timelapse', {
icon: 'timelapse',
category: 'view',
condition: isApp,
click: function () {
if (!Prop.recording) {
new Dialog({
id: 'timelapse',
title: tl('action.timelapse'),
draggable: true,
form: {
interval: {label: 'dialog.timelapse.interval', type: 'number', value: 10, step: 0.25},
source: {label: 'dialog.timelapse.source', type: 'select', value: 'preview', options: {
preview: 'data.preview',
locked: 'dialog.timelapse.source.locked',
interface: 'dialog.timelapse.source.interface',
}, condition: isApp},
destination: {label: 'dialog.timelapse.destination', type: 'folder', value: ''},
},
onConfirm: function(formData) {
Screencam.recordTimelapse(formData);
this.hide()
}
}).show();
} else {
Screencam.stopTimelapse();
}
}
})
2019-08-18 00:26:14 +08:00
new Action('screenshot_app', {
2019-07-18 00:02:07 +08:00
icon: 'icon-bb_interface',
category: 'view',
condition: isApp,
click: function () {Screencam.fullScreen()}
})
2019-08-18 00:26:14 +08:00
new Action('toggle_quad_view', {
2019-07-18 00:02:07 +08:00
icon: 'widgets',
category: 'view',
2019-08-18 00:26:14 +08:00
condition: () => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 9}),
click: function () {
main_preview.toggleFullscreen()
}
})
2020-03-05 03:56:17 +08:00
new Action('toggle_camera_projection', {
icon: 'switch_video',
2019-07-18 00:02:07 +08:00
category: 'view',
2020-03-05 03:56:17 +08:00
condition: _ => (!preview.movingBackground || !Modes.display),
keybind: new Keybind({key: 101}),
2019-07-18 00:02:07 +08:00
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.setProjectionMode(!quad_previews.current.isOrtho);
2019-07-18 00:02:07 +08:00
}
})
2020-03-05 03:56:17 +08:00
new Action('camera_initial', {
name: tl('action.load_camera_angle', tl('menu.preview.angle.initial')),
description: tl('action.load_camera_angle.desc', tl('menu.preview.angle.initial')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
2020-03-05 03:56:17 +08:00
color: 'y',
2019-07-18 00:02:07 +08:00
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2020-03-05 03:56:17 +08:00
keybind: new Keybind({key: 97}),
2019-07-18 00:02:07 +08:00
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[0])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_top', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.top')),
description: tl('action.load_camera_angle.desc', tl('direction.top')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'y',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 104}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[1])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_bottom', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.bottom')),
description: tl('action.load_camera_angle.desc', tl('direction.bottom')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'y',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 98}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[2])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_south', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.south')),
description: tl('action.load_camera_angle.desc', tl('direction.south')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'z',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 100}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[3])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_north', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.north')),
description: tl('action.load_camera_angle.desc', tl('direction.north')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'z',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 102}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[4])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_east', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.east')),
description: tl('action.load_camera_angle.desc', tl('direction.east')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'x',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 103}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[5])
2019-07-18 00:02:07 +08:00
}
})
2019-08-18 00:26:14 +08:00
new Action('camera_west', {
2020-03-05 03:56:17 +08:00
name: tl('action.load_camera_angle', tl('direction.west')),
description: tl('action.load_camera_angle.desc', tl('direction.west')),
2019-07-18 00:02:07 +08:00
icon: 'videocam',
color: 'x',
category: 'view',
2019-07-19 23:31:22 +08:00
condition: _ => !Modes.display,
2019-07-18 00:02:07 +08:00
keybind: new Keybind({key: 105}),
click: function () {
2020-03-05 03:56:17 +08:00
quad_previews.current.loadAnglePreset(DefaultCameraPresets[6])
2019-07-18 00:02:07 +08:00
}
})
})