var scene, main_preview, previews, Sun, lights, emptyMaterials, outlines, 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), b: new THREE.Color(0x2d5ee8), grid: new THREE.Color(0x495061), wire: new THREE.Color(0x576f82), outline: new THREE.Color(0x3e90ff) } 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 }, { name: 'camera_angle.isometric_right', projection: 'orthographic', position: [-64, 64*0.8165, -64], target: [0, 0, 0], zoom: 0.5, default: true }, { name: 'camera_angle.isometric_left', projection: 'orthographic', position: [64, 64*0.8165, -64], target: [0, 0, 0], zoom: 0.5, default: true } ] 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 this.angle = null; this.camPers = new THREE.PerspectiveCamera(45, 16 / 9, 1, 30000) this.camOrtho = new THREE.OrthographicCamera(-600, 600, -400, 400, -200, 20000); this.camOrtho.backgroundHandle = [{n: false, a: 'x'}, {n: false, a: 'y'}] this.camOrtho.axis = null this.camOrtho.zoom = 0.5 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; this.controls.maxDistance = 3960; this.controls.enableKeys = false; this.controls.zoomSpeed = 1.5; //Annotations this.annotations = {}; this.updateAnnotations = function() { for (var key in scope.annotations) { var tag = scope.annotations[key]; if (tag.object.visible) { var pos = tag.object.toScreenPosition(scope.camera, scope.canvas); $(tag.node).css('left', pos.x+'px'); $(tag.node).css('top', pos.y+'px'); } } } 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]; } } this.camPers.position.fromArray(DefaultCameraPresets[0].position) this.controls.target.fromArray(DefaultCameraPresets[0].target); //Keybinds this.controls.mouseButtons.ZOOM = undefined; //Renderer try { this.renderer = new THREE.WebGLRenderer({ canvas: this.canvas, antialias: true, alpha: true, preserveDrawingBuffer: true }); } catch (err) { document.querySelector('#loading_error_detail').innerHTML = 'Error creating WebGL context. Try to update your graphics drivers.'; if (isApp) { var {BrowserWindow} = require('electron').remote new BrowserWindow({ icon:'icon.ico', backgroundColor: '#ffffff', title: 'Blockbench GPU Information', webPreferences: { webgl: true, webSecurity: true, nodeIntegration: true } }).loadURL('chrome://gpu') } throw err; } this.renderer.setClearColor( 0x000000, 0 ) this.renderer.setSize(500, 400); this.loadBackground() this.selection = { box: $('
') } this.raycaster = new THREE.Raycaster() this.mouse = new THREE.Vector2(); 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) Blockbench.addDragHandler('preview_'+this.id, { extensions: ['jpg', 'jpeg', 'bmp', 'tiff', 'tif', 'gif'], element: this.canvas, readtype: 'image', }, function(files) { if (!scope.background.imgtag) { scope.background.imgtag = new Image(); } 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) { convertTouchEvent(event); var canvas_offset = $(this.canvas).offset() this.mouse.x = ((event.clientX - canvas_offset.left) / this.width) * 2 - 1; this.mouse.y = - ((event.clientY - canvas_offset.top) / this.height) * 2 + 1; this.raycaster.setFromCamera( this.mouse, this.camera ); 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) { if (intersects.length > 1 && Toolbox.selected.id == 'vertex_snap_tool') { var intersect; for (var sct of intersects) { if (sct.object.isVertex) { intersect = sct.object; break; } } if (!intersect) intersect = intersects[0].object; } else { var intersect = intersects[0].object } if (intersect.isElement) { this.controls.hasMoved = true var obj = elements.findInArray('uuid', intersects[0].object.name) 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, this.camera ) } //Camera get camera() { return this.isOrtho ? this.camOrtho : this.camPers; } setProjectionMode(ortho) { let position = this.camera.position; this.isOrtho = !!ortho; this.resize() 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; Transformer.update(); } this.setLockedAngle() this.controls.updateSceneScale(); return this; } 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); } } } 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() } Transformer.update(); this.loadBackground() return this; } 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() } if (!this.isOrtho) { this.camera.setFocalLength(preset.focal_length||45); } this.setLockedAngle(preset.locked_angle) return this; } newAnglePreset() { let scope = this; let position = scope.camera.position.toArray(); let target = scope.controls.target.toArray(); position.forEach((v, i) => { position[i] = Math.round(v*100)/100 }) target.forEach((v, i) => { target[i] = Math.round(v*100)/100 }) 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' }}, position: {label: 'dialog.save_angle.position', type: 'vector', dimensions: 3, value: position}, target: {label: 'dialog.save_angle.target', type: 'vector', dimensions: 3, value: target}, }, onConfirm: function(formResult) { if (!formResult.name) return; let preset = { name: formResult.name, projection: formResult.projection, position: formResult.position, target: formResult.target, } 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 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) { event.preventDefault(); $(':focus').blur(); unselectInterface(event); convertTouchEvent(event); this.static_rclick = event.which === 3 || event.type == 'touchstart'; if (event.type == 'touchstart') { this.rclick_cooldown = setTimeout(() => { this.rclick_cooldown = true; }, 420) } quad_previews.current = this; if (Transformer.hoverAxis !== null || (!Keybinds.extra.preview_select.keybind.isTriggered(event) && event.which !== 0)) return; var data = this.raycast(event); if (data) { //this.static_rclick = false if (data.cube && data.cube.locked) { $('#preview').css('cursor', 'not-allowed') function resetCursor() { $('#preview').css('cursor', (Toolbox.selected.cursor ? Toolbox.selected.cursor : 'default')) removeEventListeners(document, 'mouseup touchend', resetCursor, false) } addEventListeners(document, 'mouseup touchend', resetCursor, false) } else if (Toolbox.selected.selectCubes && Modes.selected.selectCubes && data.type === 'cube') { if (Toolbox.selected.selectFace) { main_uv.setFace(data.face, false) } Blockbench.dispatchEvent('canvas_select', data) if (Modes.paint) { event = 0; } if (Format.bone_rig && ( Animator.open || (!Format.rotate_cubes && ['rotate_tool', 'pivot_tool'].includes(Toolbox.selected.id)) || event.shiftKey )) { if (data.cube.parent.type === 'group') { data.cube.parent.select().showInOutliner(); } } else { data.cube.select(event) } } if (typeof Toolbox.selected.onCanvasClick === 'function') { Toolbox.selected.onCanvasClick(data) } return true; } if (typeof Toolbox.selected.onCanvasClick === 'function') { Toolbox.selected.onCanvasClick(0) } if (this.angle !== null && this.camOrtho.axis || this.movingBackground) { this.startSelRect(event) } else { return false; } } mousemove(event) { if (Settings.get('highlight_cubes')) { var data = this.raycast(event); updateCubeHighlights(data && data.cube); } } 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 } occupyTransformer(event) { 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; } if (event && event.type == 'touchstart') { Transformer.simulateMouseDown(event); } return this; } showContextMenu(event, force) { Prop.active_panel = 'preview'; if (this.static_rclick && (event.which === 3 || (event.type == 'touchend' && this.rclick_cooldown == true))) { var data = this.raycast(event) if (Toolbox.selected.selectCubes && Modes.selected.selectCubes && data && data.cube) { if (!Modes.animate) { data.cube.showContextMenu(event) } } else { this.menu.open(event, this) } } clearTimeout(this.rclick_cooldown); delete this.rclick_cooldown; return this; } //Selection Rectangle startSelRect(event) { var scope = this; if (Modes.edit || this.movingBackground) { this.sr_move_f = function(event) { scope.moveSelRect(event)} this.sr_stop_f = function(event) { scope.stopSelRect(event)} document.addEventListener('mousemove', this.sr_move_f, false) 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 this.selection.client_y = event.clientY+0 if (this.movingBackground) { this.background.before = { x: this.background.x, y: this.background.y, size: this.background.size } return }; if (!Modes.edit) return; $(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( 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), ) 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) { if ((event.shiftKey || event.ctrlOrCmd) && scope.selection.old_selected.indexOf(cube) >= 0) { var isSelected = true } else { if (cube instanceof Cube && cube.visibility && cube.mesh) { 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() } }) TickUpdates.selection = true; } stopSelRect(event) { var scope = this; document.removeEventListener('mousemove', this.sr_move_f) 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 } } else if (this.angle !== null) { this.background = canvas_scenes['ortho'+this.angle] } else { this.background = canvas_scenes.normal } return this.background } loadBackground() { this.getBackground() if (this.background && this.background.image) { if (!this.background.imgtag) this.background.imgtag = new Image(); this.background.imgtag.src = this.background.image; $(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 var zoom = (this.angle !== null && bg.lock === true) ? this.camOrtho.zoom : 1 var pos_x = 0; var pos_y = 0; if (this.angle !== null && bg.lock !== false) { 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() { if (this.movingBackground) { this.stopMovingBackground() } 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; if (this.movingBackground) { this.stopMovingBackground() } var dialog = new Dialog({ id: 'background_position', title: tl('message.set_background_position.title'), lines: [ ` ` ], 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; if (!options) options = 0; Canvas.withoutGizmos(function() { scope.render() var dataUrl = scope.canvas.toDataURL() if (options.crop == false && !options.width && !options.height) { Screencam.returnScreenshot(dataUrl, cb) } dataUrl = dataUrl.replace('data:image/png;base64,','') Jimp.read(Buffer.from(dataUrl, 'base64')).then(function(image) { if (display_mode && display_slot === 'gui' && options.crop !== false) { 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 { if (options.crop !== false) { image.autocrop([0, false]) } if (options && options.width && options.height) { image.contain(options.width, options.height) } } image.getBase64(Jimp.MIME_PNG, function(a, dataUrl){ Screencam.returnScreenshot(dataUrl, cb) }) }); }) } fullscreen() { if (quad_previews.current) { quad_previews.current.controls.stopMovement() } quad_previews.current = this; quad_previews.enabled = false; $('#preview').empty() var wrapper = $('') 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([ 'screenshot_model', {icon: 'icon-player', name: 'settings.display_skin', condition: () => (display_mode && displayReferenceObjects.active.id === 'player'), click: function() { changeDisplaySkin() }}, 'preview_checkerboard', {icon: 'wallpaper', name: 'menu.preview.background', children(preview) { var has_background = !!preview.background.image return [ {icon: 'folder', name: 'menu.preview.background.load', click: function(preview) { Blockbench.import({ resource_id: 'preview_background', 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) }}, {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(); } }}, {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', condition: (has_background && preview.background.lock !== null && preview.angle !== null), icon: preview.background.lock?'check_box':'check_box_outline_blank', click: function(preview) { preview.background.lock = !preview.background.lock preview.updateBackground() }}, {icon: 'clear', name: 'generic.remove', condition: has_background, click: function(preview) { preview.clearBackground() }} ] }}, '_', {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 = [ ] 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; }}, {icon: (preview) => (preview.isOrtho ? 'check_box' : 'check_box_outline_blank'), name: 'menu.preview.orthographic', click: function(preview) { preview.setProjectionMode(!preview.isOrtho); }}, '_', {icon: 'widgets', name: 'menu.preview.quadview', condition: function(preview) {return !quad_previews.enabled && !preview.movingBackground && !Modes.display && !Animator.open}, click: function() { openQuadView() }}, {icon: 'web_asset', name: 'menu.preview.maximize', condition: function(preview) {return quad_previews.enabled && !preview.movingBackground && !Modes.display}, click: function(preview) { 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 = $('') wrapper1.append(quad_previews.one.canvas) $('#preview').append(wrapper1) var wrapper2 = $('') wrapper2.append(quad_previews.two.canvas) $('#preview').append(wrapper2) var wrapper3 = $('') wrapper3.append(quad_previews.three.canvas) $('#preview').append(wrapper3) var wrapper4 = $('') wrapper4.append(quad_previews.four.canvas) $('#preview').append(wrapper4) updateInterface() } const Screencam = { recording_timelapse: false, fullScreen(options, cb) { setTimeout(function() { currentwindow.capturePage().then(function(screenshot) { var dataUrl = screenshot.toDataURL() if (!(options && options.width && options.height)) { Screencam.returnScreenshot(dataUrl, cb) return; } dataUrl = dataUrl.replace('data:image/png;base64,','') Jimp.read(Buffer.from(dataUrl, 'base64')).then(function(image) { image.contain(options.width, options.height) 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) { if (is_gif) { Blockbench.export({ resource_id: 'screenshot', extensions: ['gif'], type: tl('data.image'), savetype: 'binary', content: Buffer(dataUrl.split(',')[1], 'base64') }) } else { Blockbench.export({ resource_id: 'screenshot', extensions: ['png'], type: tl('data.image'), savetype: 'image', content: dataUrl }) } } else if (result === 2) { clipboard.writeImage(screenshot) } }) } else { new Dialog({ title: tl('message.screenshot.right_click'), id: 'screenie', lines: ['