2017-10-31 14:49:01 +01:00
<!DOCTYPE html>
< html >
< head >
2019-01-09 15:54:35 +01:00
< title > Blockbench< / title >
2019-07-17 18:02:07 +02:00
< meta charset = "utf-8" / >
2019-01-09 15:54:35 +01:00
< meta name = "viewport" content = "width=device-width, initial-scale=1" >
2019-07-17 18:02:07 +02:00
< meta name = "theme-color" content = "#181a1f" >
2019-04-12 18:44:18 +02:00
< meta name = "robots" content = "noindex" >
2019-07-26 13:33:29 +02:00
<!-- link rel="manifest" href="manifest.json" -->
2019-01-09 15:54:35 +01:00
< link rel = "shortcut icon" href = "favicon.png" type = "image/x-icon" / >
< link rel = "stylesheet" href = "css/w3.css" >
< link rel = "stylesheet" href = "css/jquery-ui.min.css" >
2019-07-17 18:02:07 +02:00
< link rel = "stylesheet" href = "css/fontawesome.css" >
2019-01-09 15:54:35 +01:00
< link rel = "stylesheet" href = "css/spectrum.css" >
2019-07-17 18:02:07 +02:00
< link rel = "stylesheet" href = "css/setup.css" >
< link rel = "stylesheet" href = "css/general.css" >
< link rel = "stylesheet" href = "css/panels.css" >
< link rel = "stylesheet" href = "css/dialogs.css" >
2019-01-09 15:54:35 +01:00
< style type = "text/css" id = "bbstyle" > < / style >
2017-10-31 14:49:01 +01:00
< / head >
< body spellcheck = "false" >
2019-07-17 18:02:07 +02:00
< div id = "loading_error_message" style = "display: none;" >
< div > An error occurred while loading Blockbench< / div >
< button onclick = "Blockbench.reload()" class = "large" style = "margin-right: auto; margin-left: auto;" > Reload< / button >
< button onclick = "window.close()" class = "large" style = "margin-right: auto; margin-left: auto;" > Quit< / button >
2019-01-09 15:54:35 +01:00
< / div >
< script >
if (typeof module === 'object') {window.module = module; module = undefined;}//jQuery Fix
const isApp = typeof require !== 'undefined';
2019-09-04 10:37:38 +03:00
const appVersion = '3.1.1';
2019-01-09 15:54:35 +01:00
< / script >
< script src = "lib/vue.min.js" > < / script >
< script src = "lib/vue_sortable.js" > < / script >
< script src = "lib/jquery.js" > < / script >
< script src = "lib/jquery-ui.min.js" > < / script >
< script src = "lib/targa.js" > < / script >
< script src = "lib/jimp.min.js" > < / script >
2019-03-09 22:06:35 +01:00
< script src = "lib/jszip.min.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "lib/gif.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "lib/gifshot.min.js" > < / script >
< script src = "lib/lzutf8.js" > < / script >
2019-04-07 18:53:33 +02:00
< script src = "lib/peer.min.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "lib/marked.min.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "lib/spectrum.js" > < / script >
< script src = "lib/three.js" > < / script >
< script src = "lib/three_custom.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "lib/molang.js" > < / script >
< script src = "js/preview/OrbitControls.js" > < / script >
< script src = "js/outliner/tree.vue.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/util.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "js/interface/language.js" > < / script >
< script src = "js/interface/menu.js" > < / script >
< script src = "js/interface/actions.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/blockbench.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "js/interface/keyboard.js" > < / script >
< script src = "js/interface/settings.js" > < / script >
< script src = "js/interface/dialog.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/undo.js" > < / script >
2019-04-07 18:53:33 +02:00
< script src = "js/edit_sessions.js" > < / script >
2019-01-09 15:54:35 +01:00
< script type = "text/javascript" >
if (isApp === true) {
2019-07-17 18:02:07 +02:00
document.write("< script src = 'js/desktop.js' > < \ / s c r i p t > " ) ;
2019-01-09 15:54:35 +01:00
} else {
document.write("< script src = 'js/web.js' > < \ / s c r i p t > " ) ;
}
< / script >
< script src = "js/api.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "js/outliner/outliner.js" > < / script >
< script src = "js/outliner/group.js" > < / script >
< script src = "js/outliner/cube.js" > < / script >
< script src = "js/outliner/locator.js" > < / script >
< script src = "js/preview/preview.js" > < / script >
< script src = "js/preview/canvas.js" > < / script >
< script src = "js/preview/transformer.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/transform.js" > < / script >
< script src = "js/textures.js" > < / script >
< script src = "js/uv.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "js/interface/interface.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/painter.js" > < / script >
2019-08-17 18:26:14 +02:00
< script src = "js/display_mode.js" > < / script >
2019-01-09 15:54:35 +01:00
< script src = "js/animations.js" > < / script >
< script src = "js/plugin_loader.js" > < / script >
2019-07-17 18:02:07 +02:00
< script src = "js/io/io.js" > < / script >
< script src = "js/io/bbmodel.js" > < / script >
< script src = "js/io/java_block.js" > < / script >
< script src = "js/io/bedrock.js" > < / script >
< script src = "js/io/bedrock_old.js" > < / script >
< script src = "js/io/obj.js" > < / script >
< script src = "js/io/modded_entity.js" > < / script >
2019-08-17 18:26:14 +02:00
< script src = "js/io/optifine_jem.js" > < / script >
< script src = "js/io/optifine_jpm.js" > < / script >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< script > if ( window . module ) module = window . module ; < / script >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< div style = "display: none;" > < / div >
2019-01-09 15:54:35 +01:00
2019-03-09 22:06:35 +01:00
< div id = "overlay_message_box" style = "display: none;" >
< div >
< h3 > < i class = "material-icons" > keyboard< / i > < span class = "tl" > keybindings.recording< / span > < / h3 >
< p class = "tl" > keybindings.press< / p >
2019-08-17 18:26:14 +02:00
< button class = "tl" onclick = "Keybinds.recording.stopRecording()" > dialog.cancel< / button >
< button class = "tl" onclick = "Keybinds.recording.clear().stopRecording()" > keybindings.clear< / button >
2019-03-09 22:06:35 +01:00
< div id = "keybind_input_box" contenteditable = "true" style = "font-size: 0" > < / div >
< / div >
< / div >
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "updater" >
< div class = "dialog_handle tl" > dialog.update.title< / div >
2019-01-09 15:54:35 +01:00
< h1 > < / h1 >
< div id = "updater_content" > < / div >
< div class = "progress_bar" id = "update_bar" >
< div class = "progress_bar_inner" > < / div >
< / div >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "cancel_btn confirm_btn uc_btn tl" onclick = "hideDialog()" > dialog.close< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "plugins" >
< div class = "dialog_handle tl" > dialog.plugins.title< / div >
2019-01-09 15:54:35 +01:00
< div class = "bar next_to_title" id = "plugins_header_bar" > < / div >
< div class = "bar" >
2019-08-17 18:26:14 +02:00
< div class = "tab_bar" >
< div class = "open tl" onclick = "switchPluginTabs(true)" id = "installed_plugins" > dialog.plugins.installed< / div >
< div class = "tl" onclick = "switchPluginTabs(false)" id = "all_plugins" > dialog.plugins.available< / div >
< / div >
2019-01-09 15:54:35 +01:00
< div class = "search_bar" >
< input type = "text" class = "dark_bordered" id = "plugin_search_bar" oninput = "Plugins.updateSearch()" >
< i class = "material-icons" id = "plugin_search_bar_icon" > search< / i >
< / div >
< / div >
< ul class = "list" id = "plugin_list" >
2019-02-03 21:09:35 +01:00
< li v-for = "plugin in plugin_search" v-bind:plugin = "plugin.id" v-bind:class = "{testing: plugin.fromFile, expanded: plugin.expanded}" >
< div class = "title" v-on:click = "plugin.toggleInfo()" >
2019-08-17 18:26:14 +02:00
< div class = "icon_wrapper plugin_icon normal" v-html = "Blockbench.getIconNode(plugin.icon, plugin.color).outerHTML" > < / div >
2019-01-09 15:54:35 +01:00
< i v-if = "plugin.expanded" class = "material-icons plugin_expand_icon" > expand_less< / i >
< i v-else class = "material-icons plugin_expand_icon" > expand_more< / i >
{{ plugin.title }}
< / div >
2019-02-03 21:09:35 +01:00
< div class = "button_bar" v-if = "plugin.isInstallable()" >
< button type = "button" class = "" v-on:click = "plugin.uninstall()" v-if = "plugin.installed" > < i class = "material-icons" > delete< / i > < span class = "tl" > dialog.plugins.uninstall< / span > < / button >
2019-04-07 18:53:33 +02:00
< button type = "button" class = "" v-on:click = "plugin.download(true)" v-else > < i class = "material-icons" > add< / i > < span class = "tl" > dialog.plugins.install< / span > < / button >
2019-01-09 15:54:35 +01:00
< button type = "button" class = "local_only" v-on:click = "plugin.reload()" v-if = "plugin.installed && plugin.fromFile && isApp" > < i class = "material-icons" > refresh< / i > < span class = "tl" > dialog.plugins.reload< / span > < / button >
< / div >
< div class = "button_bar tiny tl" v-else > {{ checkIfInstallable(plugin) }}< / div >
< div class = "author" > {{ tl('dialog.plugins.author', [plugin.author]) }}< / div >
< div class = "description" > {{ plugin.description }}< / div >
2019-07-17 18:02:07 +02:00
< div v-if = "plugin.expanded" class = "about" v-html = "marked(plugin.about)" > < button > a< / button > < / div >
2019-02-03 21:09:35 +01:00
< div v-if = "plugin.expanded" class = "tl" v-on:click = "plugin.toggleInfo()" style = "text-decoration: underline;" > dialog.plugins.show_less< / div >
2019-01-09 15:54:35 +01:00
< / li >
2019-02-03 21:09:35 +01:00
< div class = "no_plugin_message tl" v-if = "plugin_search.length < 1 && showAll === false" > dialog.plugins.none_installed< / div >
< div class = "no_plugin_message tl" v-if = "plugin_search.length < 1 && showAll === true" id = "plugin_available_empty" > dialog.plugins.none_available< / div >
2019-01-09 15:54:35 +01:00
< / ul >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "cancel_btn confirm_btn uc_btn tl" onclick = "saveInstalledPlugins();hideDialog();" > dialog.close< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "edit_sessions" >
< div class = "dialog_handle tl" > dialog.edit_session.title< / div >
2019-04-07 18:53:33 +02:00
< div class = "dialog_bar" >
< label class = "name_space_left tl" > edit_session.username< / label >
< input type = "text" class = "dark_bordered half" id = "edit_session_username" >
< / div >
< div class = "dialog_bar" >
< label class = "name_space_left tl" > edit_session.token< / label >
< input type = "text" class = "dark_bordered half f_left" id = "edit_session_token" >
< div id = "edit_session_copy_button" class = "tool" onclick = "EditSession.copyToken()" > < div class = "tooltip tl" > action.paste< / div > < i class = "fa fa_big fa-clipboard" > < / i > < / div >
< / div >
< div class = "edit_session_inactive" >
< p class = "tl" > edit_session.about< / p >
< p > This feature is in BETA. Bugs may occur while using it.< / p >
< / div >
< div class = "edit_session_active hidden" >
< p > < b class = "tl" > edit_session.status< / b > : < span class = "tl" id = "edit_session_status" > edit_session.connected< / span > < / p >
< / div >
< div class = "dialog_bar" >
< button type = "button" class = "edit_session_inactive confirm_btn tl" onclick = "EditSession.join();" > edit_session.join< / button >
< button type = "button" class = "edit_session_inactive tl" onclick = "EditSession.start();" > edit_session.create< / button >
< button type = "button" class = "edit_session_active tl" onclick = "EditSession.quit();" > edit_session.quit< / button >
< button type = "button" class = "cancel_btn tl" onclick = "hideDialog();" > dialog.cancel< / button >
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-04-07 18:53:33 +02:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "toolbar_edit" >
< div class = "dialog_handle tl" > dialog.toolbar_edit.title< / div >
2019-01-09 15:54:35 +01:00
< ul class = "bar" id = "bar_items_current" v-sortable = "{onChoose: choose, onUpdate: sort, onEnd: drop, animation: 160 }" >
< li v-for = "item in currentBar" v-bind:title = "item.name" :key = "item.id||item" >
< div v-if = "typeof item === 'string'" class = "toolbar_separator" > < / div >
< div v-else class = "tool" >
< div class = "tooltip" > {{item.name + (BARS.condition(item.condition) ? '' : ' (' + tl('dialog.toolbar_edit.hidden') + ')' )}}< / div >
2019-02-03 21:09:35 +01:00
< span class = "icon_wrapper" v-bind:style = "{opacity: BARS.condition(item.condition) ? 1 : 0.4}" v-html = "Blockbench.getIconNode(item.icon, item.color).outerHTML" > < / span >
2019-01-09 15:54:35 +01:00
< / div >
< / li >
< / ul >
< div class = "bar" >
< div class = "search_bar" >
< input type = "text" class = "dark_bordered" id = "action_search_bar" oninput = "BARS.list.updateSearch()" >
< i class = "material-icons" id = "plugin_search_bar_icon" > search< / i >
< / div >
< / div >
< ul class = "list" id = "bar_item_list" >
< li v-for = "item in searchedBarItems" v-on:click = "addItem(item)" >
< div class = "icon_wrapper normal" v-html = "Blockbench.getIconNode(item.icon, item.color).outerHTML" > < / div >
< div class = "icon_wrapper add" > < i class = "material-icons" > add< / i > < / div >
{{ item.name }}
< / li >
< / ul >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "cancel_btn confirm_btn uc_btn tl" onclick = "saveInstalledPlugins();hideDialog();" > dialog.close< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "entity_import" >
< div class = "dialog_handle tl" > dialog.entitylist.title< / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar narrow tl" > dialog.entitylist.text< / div >
< div class = "search_bar" >
< input type = "text" class = "dark_bordered" id = "pe_search_bar" oninput = "pe_list._data.search_text = $(this).val().toUpperCase()" >
< i class = "material-icons" id = "plugin_search_bar_icon" > search< / i >
< / div >
< ul id = "pe_list" class = "list" >
2019-07-17 18:02:07 +02:00
< li v-for = "item in searched" v-bind:class = "{ selected: item.selected }" v-on:click = "selectE(item, $event)" v-on:dblclick = "open(item)" >
2019-01-09 15:54:35 +01:00
< img class = "pe_icon" v-if = "item.icon" v-bind:src = "item.icon" >
< div class = "pe_icon" v-else > < / div >
< h4 > {{ item.title }} < span > ({{ item.name }})< / span > < / h4 >
< p > {{ item.bonecount+' '+tl('dialog.entitylist.bones') }}, {{ item.cubecount+' '+tl('dialog.entitylist.cubes') }}< / p >
< / li >
< / ul >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "tl confirm_btn" onclick = "" > dialog.import< / button >
2019-01-09 15:54:35 +01:00
< / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "image_extruder" >
< div class = "dialog_handle tl" > dialog.extrude.title< / div >
2019-01-09 15:54:35 +01:00
< h1 > < / h1 >
< div class = "dialog_bar" >
< label class = "tl" > dialog.extrude.mode< / label >
2019-07-17 18:02:07 +02:00
< div class = "bar_select f_left" >
< select id = "scan_mode" name = "scan_mode" >
< option class = "tl" id = "areas" selected > dialog.extrude.mode.areas< / option >
< option class = "tl" id = "lines" > dialog.extrude.mode.lines< / option >
< option class = "tl" id = "columns" > dialog.extrude.mode.columns< / option >
< option class = "tl" id = "pixels" > dialog.extrude.mode.pixels< / option >
< / select >
< / div >
2019-01-09 15:54:35 +01:00
< / div >
< div class = "dialog_bar" >
< label class = "tl" > dialog.extrude.opacity< / label >
< input class = "tool" type = "range" id = "scan_tolerance" value = "255" min = "1" max = "255" >
< label id = "scan_tolerance_label" > 255< / label >
< / div >
< canvas height = "256" width = "256" id = "extrusion_canvas" > < / canvas >
< div class = "dialog_bar" >
2019-07-17 18:02:07 +02:00
< button type = "button" class = "confirm_btn tl" onclick = "Extruder.startConversion()" > Scan and Import< / button >
< button type = "button" class = "cancel_btn tl" onclick = "hideDialog()" > dialog.cancel< / button >
2019-01-09 15:54:35 +01:00
< / div >
2019-07-17 18:02:07 +02:00
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "texture_edit" >
2019-08-17 18:26:14 +02:00
< div class = "dialog_handle tl" id = "te_title" > data.texture< / div >
2019-01-09 15:54:35 +01:00
< div id = "texture_menu_thumbnail" > < / div >
< div class = "bar" >
2019-02-03 21:09:35 +01:00
< div class = "tool link_only" onclick = "textures.selected.reopen()" > < div class = "tooltip tl" > menu.texture.change< / div > < i class = "material-icons" > file_upload< / i > < / div >
< div class = "tool link_only" onclick = "textures.selected.refresh(true)" > < div class = "tooltip tl" > menu.texture.refresh< / div > < i class = "material-icons" > refresh< / i > < / div >
< div class = "tool link_only" onclick = "textures.selected.openFolder()" > < div class = "tooltip tl" > menu.texture.folder< / div > < i class = "material-icons" > folder< / i > < / div >
2019-03-11 19:57:22 +01:00
< div class = "tool" onclick = "textures.selected.remove()" > < div class = "tooltip tl" > generic.delete< / div > < i class = "material-icons" > delete< / i > < / div >
2019-01-09 15:54:35 +01:00
< / div >
< p class = "multiline_text" id = "te_path" > path< / p >
2019-08-17 18:26:14 +02:00
< div class = "dialog_bar narrow bitmap_only" > < label class = "tl" > generic.name< / label > < / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar bitmap_only" >
< input type = "text" class = "input_wide dark_bordered" id = "te_name" >
< / div >
< div class = "dialog_bar narrow" > < label class = "tl" > dialog.texture.variable< / label > < / div >
< div class = "dialog_bar" >
< input type = "text" class = "input_wide dark_bordered" id = "te_variable" >
< / div >
< div class = "dialog_bar narrow" > < label class = "tl" > dialog.texture.folder< / label > < / div >
< div class = "dialog_bar" >
< input type = "text" class = "input_wide dark_bordered" id = "te_folder" >
< / div >
< div class = "dialog_bar narrow" > < label class = "tl" > dialog.texture.namespace< / label > < / div >
< div class = "dialog_bar" >
< input type = "text" class = "input_wide dark_bordered" id = "te_namespace" >
< / div >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "confirm_btn cancel_btn" onclick = "saveTextureMenu()" > Close< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "scaling" >
< div class = "dialog_handle tl" > dialog.scale.title< / div >
2019-01-09 15:54:35 +01:00
2019-04-07 18:53:33 +02:00
< label class = "tl" > dialog.scale.axis< / label >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" style = "height: 32px;" >
< input type = "checkbox" class = "toggle_panel" id = "model_scale_x_axis" onchange = "scaleAll()" checked >
< label class = "toggle_panel" for = "model_scale_x_axis" > X< / label >
< input type = "checkbox" class = "toggle_panel" id = "model_scale_y_axis" onchange = "scaleAll()" checked >
< label class = "toggle_panel" for = "model_scale_y_axis" > Y< / label >
< input type = "checkbox" class = "toggle_panel" id = "model_scale_z_axis" onchange = "scaleAll()" checked >
< label class = "toggle_panel" for = "model_scale_z_axis" > Z< / label >
< / div >
2019-04-07 18:53:33 +02:00
< label class = "tl" > data.origin< / label >
< div class = "dialog_bar" >
< label for = "scaling_origin_x" class = "inline_label tl" > X< / label >
2019-07-19 17:31:22 +02:00
< input type = "number" id = "scaling_origin_x" class = "dark_bordered medium_width" oninput = "scaleAll()" >
2019-04-07 18:53:33 +02:00
< label for = "scaling_origin_y" class = "inline_label tl" > Y< / label >
2019-07-19 17:31:22 +02:00
< input type = "number" id = "scaling_origin_y" class = "dark_bordered medium_width" oninput = "scaleAll()" >
2019-04-07 18:53:33 +02:00
< label for = "scaling_origin_z" class = "inline_label tl" > Z< / label >
2019-07-19 17:31:22 +02:00
< input type = "number" id = "scaling_origin_z" class = "dark_bordered medium_width" oninput = "scaleAll()" >
2019-01-09 15:54:35 +01:00
< / div >
2019-04-07 18:53:33 +02:00
< label class = "tl" > dialog.scale.scale< / label >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" style = "height: 32px;" >
< input type = "range" id = "model_scale_range" value = "1" min = "0" max = "4" step = "0.02" oninput = "modelScaleSync()" >
< input type = "number" class = "f_left" id = "model_scale_label" min = "0" max = "4" step = "0.02" value = "1" oninput = "modelScaleSync(true)" >
< / div >
2019-04-07 18:53:33 +02:00
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar narrow" id = "scaling_clipping_warning" > < / div >
< div class = "dialog_bar" >
2019-04-07 18:53:33 +02:00
< button type = "button" onclick = "scaleAll(true)" class = "confirm_btn tl" > dialog.scale.confirm< / button >
< button type = "button" class = "cancel_btn tl" onclick = "cancelScaleAll()" > dialog.cancel< / button >
2019-04-12 18:44:18 +02:00
< button type = "button" class = "minor hidden tl" id = "scale_overflow_btn" onclick = "scaleAllSelectOverflow()" > dialog.scale.select_overflow< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "create_preset" >
< div class = "dialog_handle tl" > dialog.display_preset.title< / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar tl" > dialog.display_preset.message< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "thirdperson_righthand_save" checked >
< label for = "thirdperson_righthand_save" class = "tl" > display.slot.third_right< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "thirdperson_lefthand_save" checked >
< label for = "thirdperson_lefthand_save" class = "tl" > display.slot.third_left< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "firstperson_righthand_save" checked >
< label for = "firstperson_righthand_save" class = "tl" > display.slot.first_right< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "firstperson_lefthand_save" checked >
< label for = "firstperson_lefthand_save" class = "tl" > display.slot.first_left< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "head_save" checked >
< label for = "head_save" class = "tl" > display.slot.head< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "ground_save" checked >
< label for = "ground_save" class = "tl" > display.slot.ground< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "fixed_save" checked >
< label for = "fixed_save" class = "tl" > display.slot.frame< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "gui_save" checked >
< label for = "gui_save" class = "tl" > display.slot.gui< / label >
< / div >
< div class = "dialog_bar narrow" >
2019-08-17 18:26:14 +02:00
< label class = "tl" > generic.name< / label >
2019-01-09 15:54:35 +01:00
< / div >
< div class = "dialog_bar" >
< input type = "text" id = "preset_name" class = "input_wide" id = "new preset" >
< / div >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "tl confirm_btn" onclick = "DisplayMode.createPreset()" > dialog.display_preset.create< / button >
< button type = "button" class = "tl cancel_btn" onclick = "hideDialog()" > dialog.cancel< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "selection_creator" >
< div class = "dialog_handle tl" > dialog.select.title< / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" >
< input type = "checkbox" id = "selgen_new" checked >
< label class = "name_space_left tl" for = "selgen_new" > dialog.select.new< / label >
< / div >
< div class = "dialog_bar" >
< input type = "checkbox" id = "selgen_group" >
< label class = "name_space_left tl" for = "selgen_group" > dialog.select.group< / label >
< / div >
< div class = "dialog_bar" >
< label class = "name_space_left tl" > dialog.select.name< / label >
< input type = "text" class = "dark_bordered half" id = "selgen_name" >
< / div >
2019-08-17 18:26:14 +02:00
< div class = "dialog_bar" >
< label class = "name_space_left tl" > data.texture< / label >
< input type = "text" class = "dark_bordered half" id = "selgen_texture" >
< / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" >
< label class = "name_space_left tl" > dialog.select.random< / label >
< input type = "range" min = "0" max = "100" step = "1" value = "100" class = "tool half" id = "selgen_random" >
< / div >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "tl confirm_btn" onclick = "createSelection()" > dialog.select.select< / button >
< button type = "button" class = "tl cancel_btn" onclick = "hideDialog()" > dialog.cancel< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "settings" >
< div class = "dialog_handle tl" > dialog.settings.settings< / div >
2019-08-17 18:26:14 +02:00
< div class = "dialog_bar borderless tab_bar" id = "settings_tab_bar" >
2019-01-09 15:54:35 +01:00
< div class = "tl tab open" id = "setting" onclick = "setSettingsTab('setting')" > dialog.settings.settings< / div >
< div class = "tl tab" id = "keybindings" onclick = "setSettingsTab('keybindings')" > dialog.settings.keybinds< / div >
< div class = "tl tab" id = "layout_settings" onclick = "setSettingsTab('layout_settings')" > dialog.settings.layout< / div >
< div class = "tl tab" id = "credits" onclick = "setSettingsTab('credits')" > dialog.settings.about< / div >
< / div >
2019-04-12 18:44:18 +02:00
2019-01-09 15:54:35 +01:00
< div id = "setting" class = "tab_content" >
< h2 class = "tl i_b" > dialog.settings.settings< / h2 >
2019-07-17 18:02:07 +02:00
< div class = "search_bar" >
< input type = "text" class = "dark_bordered" id = "settings_search_bar" oninput = "Settings.updateSearch()" >
< i class = "material-icons" id = "settings_search_bar_icon" > search< / i >
< / div >
2019-01-09 15:54:35 +01:00
< ul id = "settingslist" >
2019-07-17 18:02:07 +02:00
< li v-for = "category in structure" v-if = "!category.hidden" >
2019-01-09 15:54:35 +01:00
< h3 v-on:click = "toggleCategory(category)" >
< i class = "material-icons" > {{ category.open ? 'expand_more' : 'navigate_next' }}< / i >
{{ category.name }}
< / h3 >
< ul v-if = "category.open" >
< li v-for = "(setting, key) in category.items" v-if = "Condition(setting.condition)" v-on = "setting.click ? {click: setting.click} : {}" >
< template v-if = "setting.type === 'number'" >
< div class = "setting_element" > < input type = "number" v-model . number = "setting.value" v-on:input = "saveSettings()" > < / div >
< / template >
< template v-else-if = "setting.type === 'click'" >
2019-07-17 18:02:07 +02:00
< div class = "setting_element setting_icon" v-html = "Blockbench.getIconNode(setting.icon).outerHTML" > < / div >
2019-01-09 15:54:35 +01:00
< / template >
< template v-else-if = "!setting.type" > <!-- TOGGLE -->
< div class = "setting_element" > < input type = "checkbox" v-model = "setting.value" v-bind:id = "'setting_'+key" v-on:click = "saveSettings()" > < / div >
< / template >
< label class = "setting_label" v-bind:for = "'setting_'+key" >
< div class = "setting_name" > {{ tl('settings.'+key) }}< / div >
< div class = "setting_description" > {{ tl('settings.'+key+'.desc') }}< / div >
< / label >
< template v-if = "setting.type === 'text'" >
< input type = "text" class = "dark_bordered" style = "width: 96%" v-model = "setting.value" v-on:input = "saveSettings()" >
< / template >
< template v-else-if = "setting.type === 'select'" >
2019-04-07 18:53:33 +02:00
< div class = "bar_select" >
< select v-model = "setting.value" >
< option v-for = "(text, id) in setting.options" v-bind:value = "id" > {{ text }}< / option >
< / select >
< / div >
2019-01-09 15:54:35 +01:00
< / template >
< / li >
< / ul >
< / li >
< / ul >
< / div >
2019-04-12 18:44:18 +02:00
2019-01-09 15:54:35 +01:00
< div id = "keybindings" class = "hidden tab_content" >
< h2 class = "tl i_b" > dialog.settings.keybinds< / h2 >
< div class = "bar next_to_title" id = "keybinds_title_bar" > < / div >
2019-07-17 18:02:07 +02:00
< div class = "search_bar" >
< input type = "text" class = "dark_bordered" id = "keybind_search_bar" oninput = "Keybinds.updateSearch()" >
< i class = "material-icons" id = "keybind_search_bar_icon" > search< / i >
< / div >
2019-01-09 15:54:35 +01:00
< ul id = "keybindlist" >
2019-07-17 18:02:07 +02:00
< li v-for = "category in structure" v-if = "!category.hidden" >
2019-01-09 15:54:35 +01:00
< h3 v-on:click = "toggleCategory(category)" > < i class = "material-icons" > {{ category.open ? 'expand_more' : 'navigate_next' }}< / i > {{ category.name }}< / h3 >
< ul v-if = "category.open" >
< li v-for = "action in category.actions" >
2019-07-17 18:02:07 +02:00
< div v-bind:title = "action.description" > {{action.name}}< / div >
2019-01-09 15:54:35 +01:00
< div class = "keybindslot" v-on:click . stop = "record(action)" > {{ action.keybind ? action.keybind.label : '' }}< / div >
< div class = "tool" v-on:click = "reset(action)" >
< div class = "tooltip tl" > keybindings.reset< / div >
2019-02-03 21:09:35 +01:00
< i class = "material-icons" > replay< / i >
2019-01-09 15:54:35 +01:00
< / div >
< div class = "tool" v-on:click = "clear(action)" >
< div class = "tooltip tl" > keybindings.clear< / div >
2019-02-03 21:09:35 +01:00
< i class = "material-icons" > clear< / i >
2019-01-09 15:54:35 +01:00
< / div >
< / li >
< / ul >
< / li >
< / ul >
< / div >
2019-04-12 18:44:18 +02:00
2019-01-09 15:54:35 +01:00
< div id = "layout_settings" class = "hidden tab_content" >
< h2 class = "tl i_b" > dialog.settings.layout< / h2 >
< div class = "bar next_to_title" id = "layout_title_bar" > < / div >
2019-07-19 17:31:22 +02:00
< div class = "y_scrollable" id = "layout_list" >
< div id = "color_wrapper" >
2019-01-09 15:54:35 +01:00
< / div >
2019-07-19 17:31:22 +02:00
< div class = "dialog_bar" >
< label class = "name_space_left tl" for = "layout_font_main" > layout.font.main< / label >
< input type = "text" class = "half dark_bordered" id = "layout_font_main" oninput = "changeUIFont('main')" >
< / div >
2019-01-09 15:54:35 +01:00
2019-07-19 17:31:22 +02:00
< div class = "dialog_bar" >
< label class = "name_space_left tl" for = "layout_font_headline" > layout.font.headline< / label >
< input type = "text" class = "half dark_bordered" id = "layout_font_headline" oninput = "changeUIFont('headline')" >
< / div >
2019-01-09 15:54:35 +01:00
< / div >
< / div >
< div id = "credits" class = "hidden tab_content" >
< h2 class = "tl i_b" > dialog.settings.about< / h2 >
2019-07-17 18:02:07 +02:00
< div id = "about_page_title" >
< i class = "icon-blockbench_inverted" > < / i >
< span > Blockbench< / span >
< / div >
2019-01-09 15:54:35 +01:00
< p > < b class = "tl" > about.version< / b > < span id = "version_tag" > < script >
$('#version_tag').text(appVersion)
< / script > < / span > < / p >
< p > < b class = "tl" > about.creator< / b > JannisX11< / p >
< p > < b class = "tl" > about.website< / b > < a class = "open-in-browser" href = "http://blockbench.net" > blockbench.net< / a > < / p >
< p > < b class = "tl" > about.bugtracker< / b > < a class = "open-in-browser" href = "https://github.com/JannisX11/blockbench/issues" > github.com/JannisX11/blockbench< / a > < / p >
< p class = "local_only tl" > about.electron< / p >
< p class = "tl" > about.vertex_snap< / p >
< p > < b class = "tl" > about.icons< / b > < a href = "https://material.io/icons/" class = "open-in-browser" > material.io/icons< / a > & < a href = "http://fontawesome.io/icons/" class = "open-in-browser" > fontawesome< / a > < / p >
< p > < b class = "tl" > about.libraries< / b >
< a class = "open-in-browser" href = "https://jquery.com" > jQuery< / a > ,
< a class = "open-in-browser" href = "https://jqueryui.com" > jQuery UI< / a > ,
< a class = "open-in-browser" href = "http://touchpunch.furf.com" > jQuery UI Touch Punch< / a > ,
< a class = "open-in-browser" href = "https://vuejs.org" > VueJS< / a > ,
< a class = "open-in-browser" href = "https://github.com/weibangtuo/vue-tree" > Vue Tree< / a > ,
< a class = "open-in-browser" href = "https://github.com/sagalbot/vue-sortable" > Vue Sortable< / a > ,
< a class = "open-in-browser" href = "https://threejs.org" > ThreeJS< / a > ,
< a class = "open-in-browser" href = "https://github.com/oliver-moran/jimp" > Jimp< / a > ,
< a class = "open-in-browser" href = "https://bgrins.github.io/spectrum" > Spectrum< / a > ,
2019-07-17 18:02:07 +02:00
< a class = "open-in-browser" href = "https://github.com/jnordberg/gif.js" > gif.js< / a > ,
< a class = "open-in-browser" href = "https://stuk.github.io/jszip/" > JSZip< / a > ,
2019-09-06 00:16:54 +02:00
< a class = "open-in-browser" href = "https://github.com/rotemdan/lzutf8.js" > LZ-UTF8< / a > ,
2019-07-17 18:02:07 +02:00
< a class = "open-in-browser" href = "https://github.com/markedjs/marked" > Marked< / a >
2019-01-09 15:54:35 +01:00
< / p >
< / div >
2019-04-12 18:44:18 +02:00
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "confirm_btn cancel_btn tl" onclick = "Settings.save()" > dialog.close< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable" id = "uv_dialog" >
< div class = "dialog_handle tl" > uv_editor.title< / div >
2019-08-17 18:26:14 +02:00
< div class = "dialog_bar borderless tab_bar" id = "uv_tab_bar" >
2019-01-09 15:54:35 +01:00
< div onclick = "uv_dialog.openTab('all')" id = "all" class = "tab open tl" > uv_editor.all_faces< / div >
< div onclick = "uv_dialog.openTab('north')" id = "north" class = "tab tl" > face.north< / div >
< div onclick = "uv_dialog.openTab('south')" id = "south" class = "tab tl" > face.south< / div >
< div onclick = "uv_dialog.openTab('west')" id = "west" class = "tab tl" > face.west< / div >
< div onclick = "uv_dialog.openTab('east')" id = "east" class = "tab tl" > face.east< / div >
< div onclick = "uv_dialog.openTab('up')" id = "up" class = "tab tl" > face.up< / div >
< div onclick = "uv_dialog.openTab('down')" id = "down" class = "tab tl" > face.down< / div >
< / div >
< div id = "uv_dialog_all" class = "uv_dialog_content uv_dialog_all_only" >
< / div >
2019-07-17 18:02:07 +02:00
2019-01-09 15:54:35 +01:00
< div id = "uv_dialog_single" class = "uv_dialog_content" >
< / div >
2019-07-17 18:02:07 +02:00
< div class = "bar" id = "uv_dialog_toolbar" >
2019-01-09 15:54:35 +01:00
< div class = "toolbar_wrapper uv_dialog" > < / div >
< / div >
2019-07-17 18:02:07 +02:00
2019-08-17 18:26:14 +02:00
< button type = "button" onclick = "hideDialog()" class = "confirm_btn cancel_btn hidden" > dialog.close< / button >
2019-01-09 15:54:35 +01:00
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
2019-07-17 18:02:07 +02:00
< dialog class = "dialog draggable paddinged" id = "text_input" >
< div class = "dialog_handle tl" > dialog.input.title< / div >
2019-01-09 15:54:35 +01:00
< div class = "dialog_bar" >
< input type = "text" id = "text_input_field" class = "dark_bordered input_wide" >
< / div >
< div class = "dialog_bar" >
2019-08-17 18:26:14 +02:00
< button type = "button" class = "confirm_btn tl" onclick = "hideDialog()" > dialog.confirm< / button >
< button type = "button" class = "cancel_btn tl" onclick = "hideDialog()" > dialog.cancel< / button >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "dialog_close_button" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < i class = "material-icons" > clear< / i > < / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-01-09 15:54:35 +01:00
< div id = "plugin_dialog_wrapper" > < / div >
2019-07-17 18:02:07 +02:00
< dialog id = "action_selector" v-if = "open" >
< input type = "text" v-model = "search_input" autocomplete = "off" autosave = "off" autocorrect = "off" spellcheck = "off" autocapitalize = "off" >
2019-02-03 21:09:35 +01:00
< i class = "material-icons" id = "action_search_bar_icon" > search< / i >
2019-04-07 18:53:33 +02:00
< div >
< ul >
< li v-for = "(item, i) in actions" v-on:click = "ActionControl.click(item, $event)" v-bind:class = "{selected: i === index}" v-on:mouseenter = "index = i" >
< div class = "icon_wrapper normal" v-html = "Blockbench.getIconNode(item.icon, item.color).outerHTML" > < / div >
2019-08-17 18:26:14 +02:00
< div class = "name" > {{ item.name }}< / div >
< label > {{ item.keybind.label }}< / label >
2019-04-07 18:53:33 +02:00
< / li >
< / ul >
2019-04-12 18:44:18 +02:00
< div class = "small_text" v-if = "actions[index]" > {{ Pressing.alt ? actions[index].keybind.label : actions[index].description }}< / div >
2019-04-07 18:53:33 +02:00
< / div >
2019-07-17 18:02:07 +02:00
< / dialog >
2019-02-03 21:09:35 +01:00
2019-01-09 15:54:35 +01:00
< header >
2019-07-17 18:02:07 +02:00
< ul id = "mac_window_menu" hidden > < / ul >
< div id = "title" class = "app-drag-region" >
2019-01-09 15:54:35 +01:00
< span > Blockbench< / span >
< i class = "icon-blockbench_inverted" > < / i >
< / div >
2019-07-17 18:02:07 +02:00
< ul id = "menu_bar" class = "scroll_horizontal" > < / ul >
< div class = "app-drag-region" id = "header_free_bar" > < / div >
< ul id = "windows_window_menu" hidden >
< li onclick = "currentwindow.minimize()" > < i class = "fas fa-window-minimize" > < / i > < / li >
< li onclick = "currentwindow.isMaximized() ? currentwindow.unmaximize() : currentwindow.maximize()" > < i class = "far fa-window-restore" > < / i > < / li >
< li class = "wwm_r" onclick = "currentwindow.close()" > < i class = "fas fa-times" > < / i > < / li >
< / ul >
< button id = "web_download_button" hidden > < a class = "tl" href = "https://blockbench.net/downloads" > web.download_app< / a > < / button >
< / header >
< div id = "page_wrapper" >
< div id = "blackout" onclick = "$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()" > < / div >
< div id = "main_toolbar" >
2019-08-01 11:07:33 +02:00
< div style = "color: #fff; text-align: center; font-family: segoe ui, sans-serif;" id = "no_css_message" >
< br > < h1 > Oops...< / h1 >
< p > It looks like your internet connection was too unstable to automatically update Blockbench.< / p >
< p > Download and run < a style = "color: #75b1ff" href = "https://blockbench.net/downloads/" > the installer< / a > to update manually. Your custom settings will remain untouched.< / p >
< / div >
2019-01-09 15:54:35 +01:00
< div class = "toolbar_wrapper narrow tools" > < / div >
2019-07-17 18:02:07 +02:00
< div class = "toolbar_wrapper narrow mobile_side" > < / div >
2019-01-09 15:54:35 +01:00
< div class = "toolbar_wrapper narrow tool_options" > < / div >
< div id = "mode_selector" >
< li
v-for="mode in options"
v-if="Condition(mode.condition)"
v-bind:class="{selected: mode.selected}"
v-on:click="mode.select()"
>{{ mode.name }}< / li >
< / div >
2019-07-17 18:02:07 +02:00
< /div id="main_toolbar">
2019-01-09 15:54:35 +01:00
< div id = "left_bar" class = "sidebar" >
< div id = "uv" class = "panel selection_only" >
< div class = "bar next_to_title" id = "uv_title_bar" > < / div >
2019-07-17 18:02:07 +02:00
< div id = "uv_panel_sides" onclick = "main_uv.loadSelectedFace()" class = "bar tabs_small block_mode_only" >
2019-01-09 15:54:35 +01:00
< input type = "radio" name = "side" id = "north_radio" checked >
< label class = "tl" for = "north_radio" > face.north< / label >
< input type = "radio" name = "side" id = "south_radio" >
< label class = "tl" for = "south_radio" > face.south< / label >
< input type = "radio" name = "side" id = "west_radio" >
< label class = "tl" for = "west_radio" > face.west< / label >
< input type = "radio" name = "side" id = "east_radio" >
< label class = "tl" for = "east_radio" > face.east< / label >
< input type = "radio" name = "side" id = "up_radio" >
< label class = "tl" for = "up_radio" > face.up< / label >
< input type = "radio" name = "side" id = "down_radio" >
< label class = "tl" for = "down_radio" > face.down< / label >
< / div >
< / div >
< div id = "display" class = "panel" >
< div class = "toolbar_wrapper display" > < / div >
< p class = "tl" > display.slot< / p >
< div id = "display_bar" class = "bar tabs_small" >
< input class = "hidden" type = "radio" name = "display" id = "thirdperson_righthand" checked >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "thirdperson_righthand" onclick = "DisplayMode.loadThirdRight()" > < div class = "tooltip tl" > display.slot.third_right< / div > < i class = "material-icons" > accessibility< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "thirdperson_lefthand" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "thirdperson_lefthand" onclick = "DisplayMode.loadThirdLeft()" > < div class = "tooltip tl" > display.slot.third_left< / div > < i class = "material-icons" > accessibility< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "firstperson_righthand" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "firstperson_righthand" onclick = "DisplayMode.loadFirstRight()" > < div class = "tooltip tl" > display.slot.first_right< / div > < i class = "material-icons" > person< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "firstperson_lefthand" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "firstperson_lefthand" onclick = "DisplayMode.loadFirstLeft()" > < div class = "tooltip tl" > display.slot.first_left< / div > < i class = "material-icons" > person< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "head" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "head" onclick = "DisplayMode.loadHead()" > < div class = "tooltip tl" > display.slot.head< / div > < i class = "material-icons" > sentiment_satisfied< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "ground" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "ground" onclick = "DisplayMode.loadGround()" > < div class = "tooltip tl" > display.slot.ground< / div > < i class = "icon-ground" > < / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "fixed" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "fixed" onclick = "DisplayMode.loadFixed()" > < div class = "tooltip tl" > display.slot.frame< / div > < i class = "material-icons" > filter_frames< / i > < / label >
2019-01-09 15:54:35 +01:00
< input class = "hidden" type = "radio" name = "display" id = "gui" >
2019-02-03 21:09:35 +01:00
< label class = "tool" for = "gui" onclick = "DisplayMode.loadGUI()" > < div class = "tooltip tl" > display.slot.gui< / div > < i class = "material-icons" > border_style< / i > < / label >
2019-01-09 15:54:35 +01:00
< / div >
< p class = "reference_model_bar tl" > display.reference< / p >
< div id = "display_ref_bar" class = "bar tabs_small reference_model_bar" >
< / div >
< div id = "display_sliders" >
< p class = "tl" > display.rotation< / p > < div class = "tool head_right" v-on:click = "resetChannel('rotation')" > < i class = "material-icons" > replay< / i > < / div >
< div class = "bar" v-for = "axis in [0, 1, 2]" >
< input type = "range" class = "tool disp_range" v-model . number = "slot.rotation[axis]" v-bind:trigger_type = "'rotation.'+axis" min = "-180" max = "180" step = "1" value = "0" @ input = "change" @ input = "change(axis, 'rotation')" @ mousedown = "start" @ change = "save" >
2019-07-17 18:02:07 +02:00
< input lang = "en" type = "number" class = "tool disp_text" v-model . number = "slot.rotation[axis]" min = "-180" max = "180" step = "0.5" value = "0" @ input = "change(axis, 'rotation');save()" @ mousedown = "start" >
2019-01-09 15:54:35 +01:00
< / div >
< p class = "tl" > display.translation< / p > < div class = "tool head_right" v-on:click = "resetChannel('translation')" > < i class = "material-icons" > replay< / i > < / div >
< div class = "bar" v-for = "axis in [0, 1, 2]" >
< input type = "range" class = "tool disp_range" v-model . number = "slot.translation[axis]" v-bind:trigger_type = "'translation.'+axis"
v-bind:min="Math.abs(slot.translation[axis]) < 10 ? -20 : ( slot . translation [ axis ] > 0 ? -70*3+10 : -80)"
v-bind:max="Math.abs(slot.translation[axis]) < 10 ? 20 : ( slot . translation [ axis ] < 0 ? 70 * 3-10 : 80 ) "
v-bind:step="Math.abs(slot.translation[axis]) < 10 ? 0 . 25 : 1 "
value="0" @input="change(axis, 'translation')" @mousedown="start" @change="save">
2019-07-17 18:02:07 +02:00
< input lang = "en" type = "number" class = "tool disp_text" v-model . number = "slot.translation[axis]" min = "-80" max = "80" step = "0.5" value = "0" @ input = "change(axis, 'translation');save()" @ mousedown = "start" >
2019-01-09 15:54:35 +01:00
< / div >
< p class = "tl" > display.scale< / p > < div class = "tool head_right" v-on:click = "resetChannel('scale')" > < i class = "material-icons" > replay< / i > < / div >
< div class = "bar" v-for = "axis in [0, 1, 2]" >
< div class = "tool display_scale_invert" v-on:click = "invert(axis)" >
2019-02-03 21:09:35 +01:00
< div class = "tooltip tl" > display.mirror< / div >
< i class = "material-icons" > {{ slot.mirror[axis] ? 'check_box' : 'check_box_outline_blank' }}< / i >
2019-01-09 15:54:35 +01:00
< / div >
< input type = "range" class = "tool disp_range scaleRange" v-model . number = "slot.scale[axis]" v-bind:trigger_type = "'scale.'+axis" v-bind:id = "'scale_range_'+axis"
v-bind:min="slot.scale[axis] > 1 ? -2 : 0"
v-bind:max="slot.scale[axis] > 1 ? 4 : 2"
step="0.01"
value="0" @input="change(axis, 'scale')" @mousedown="start" @change="save">
< input type = "number" class = "tool disp_text" v-model . number = "slot.scale[axis]" min = "0" max = "4" step = "0.01" value = "0" @ input = "change(axis, 'scale');save()" @ mousedown = "start" >
< / div >
< / div >
< / div >
< div id = "animations" class = "panel" >
< div class = "toolbar_wrapper animations" > < / div >
2019-08-17 18:26:14 +02:00
< ul id = "animations_list" class = "list" v-sortable = "{onUpdate: sort, onChoose: choose, fallbackTolerance: 10, animation: 0, handle: ':not(.animation_play_toggle)'}" >
2019-01-09 15:54:35 +01:00
< li
v-for="animation in animations"
v-bind:class="{ selected: animation.selected }"
v-bind:anim_id="animation.uuid"
class="animation"
v-on:click.stop="animation.select()"
2019-09-04 10:37:38 +03:00
v-on:dblclick.stop="animation.rename()"
2019-08-17 18:26:14 +02:00
:key="animation.uuid"
2019-01-09 15:54:35 +01:00
@contextmenu.prevent.stop="animation.showContextMenu($event)"
>
< i class = "material-icons" > movie< / i >
< input class = "animation_name" v-model = "animation.name" disabled = "true" >
2019-07-17 18:02:07 +02:00
< div class = "animation_play_toggle" v-on:click . stop = "animation.togglePlayingState()" >
< i v-if = "animation.playing" class = "fa_big far fa-play-circle" > < / i >
< i v-else class = "fa_big far fa-circle" > < / i >
< / div >
2019-01-09 15:54:35 +01:00
< / li >
< / ul >
< / div >
< div id = "keyframe" class = "panel" >
< div class = "toolbar_wrapper keyframe" > < / div >
< p class = "tl" id = "keyframe_type_label" > < / p >
2019-08-17 18:26:14 +02:00
< div class = "bar flex" id = "keyframe_bar_x" >
2019-01-09 15:54:35 +01:00
< label > X< / label >
< input type = "text" id = "keyframe_x" class = "dark_bordered code keyframe_input" axis = "x" oninput = "updateKeyframeValue(this)" >
< / div >
2019-08-17 18:26:14 +02:00
< div class = "bar flex" id = "keyframe_bar_y" >
2019-01-09 15:54:35 +01:00
< label > Y< / label >
< input type = "text" id = "keyframe_y" class = "dark_bordered code keyframe_input" axis = "y" oninput = "updateKeyframeValue(this)" >
< / div >
2019-08-17 18:26:14 +02:00
< div class = "bar flex" id = "keyframe_bar_z" >
2019-01-09 15:54:35 +01:00
< label > Z< / label >
< input type = "text" id = "keyframe_z" class = "dark_bordered code keyframe_input" axis = "z" oninput = "updateKeyframeValue(this)" >
< / div >
2019-08-17 18:26:14 +02:00
< div class = "bar flex" id = "keyframe_bar_w" >
2019-01-09 15:54:35 +01:00
< label > W< / label >
< input type = "text" id = "keyframe_w" class = "dark_bordered code keyframe_input" axis = "w" oninput = "updateKeyframeValue(this)" >
< / div >
2019-08-17 18:26:14 +02:00
< div class = "bar flex" id = "keyframe_bar_effect" >
< label class = "tl" > data.effect< / label >
< input type = "text" id = "keyframe_effect" class = "dark_bordered code keyframe_input" axis = "effect" oninput = "updateKeyframeValue(this)" >
< / div >
< div class = "bar flex" id = "keyframe_bar_locator" >
< label class = "tl" > data.locator< / label >
< input type = "text" id = "keyframe_locator" class = "dark_bordered code keyframe_input" axis = "locator" oninput = "updateKeyframeValue(this)" >
< / div >
2019-01-09 15:54:35 +01:00
< / div >
< div id = "variable_placeholders" class = "panel grow" >
< p class = "tl" > panel.variable_placeholders.info< / p >
< textarea id = "var_placeholder_area" class = "code" style = "flex-grow: 1;" onkeyup = "Animator.preview()" > < / textarea >
< / div >
< div id = "textures" class = "panel grow" >
< div class = "toolbar_wrapper texturelist" > < / div >
< ul id = "texture_list" class = "list" >
< li
v-for="texture in textures"
v-bind:class="{ selected: texture.selected, particle: texture.particle}"
2019-02-03 21:09:35 +01:00
v-bind:texid="texture.uuid"
2019-01-09 15:54:35 +01:00
class="texture"
v-on:click.stop="texture.select($event)"
v-on:dblclick="texture.openMenu($event)"
@contextmenu.prevent.stop="texture.showContextMenu($event)"
>
< div class = "texture_icon_wrapper" >
2019-07-17 18:02:07 +02:00
< img v-bind:texid = "texture.id" v-bind:src = "texture.source" class = "texture_icon" width = "48px" alt = "" v-if = "texture.show_icon" / >
2019-04-07 18:53:33 +02:00
< i class = "material-icons texture_error" v-bind:title = "texture.getErrorMessage()" v-if = "texture.error" > error_outline< / i >
2019-07-17 18:02:07 +02:00
< i class = "texture_movie fa fa_big fa-film" title = "Animated Texture" v-if = "texture.frameCount > 1 && !Format.single_texture" > < / i >
2019-01-09 15:54:35 +01:00
< / div >
< i class = "material-icons texture_save_icon" v-bind:class = "{clickable: !texture.saved}" v-on:click = "texture.save()" >
< template v-if = "texture.saved" > check_circle< / template >
< template v-else > save< / template >
< / i >
2019-03-11 19:57:22 +01:00
< i class = "material-icons texture_particle_icon" v-if = "texture.particle" > bubble_chart< / i >
2019-03-09 22:06:35 +01:00
< div class = "texture_name" > {{ texture.name }}< / div >
2019-07-17 18:02:07 +02:00
< div class = "texture_res" > {{ texture.error
? texture.getErrorMessage()
2019-09-06 00:16:54 +02:00
: texture.width + ' x ' + texture.height + 'px'
2019-07-17 18:02:07 +02:00
}}< / div >
2019-01-09 15:54:35 +01:00
< / li >
< / ul >
< / div >
< / div >
< div id = "right_bar" class = "sidebar" >
2019-07-19 17:31:22 +02:00
< div id = "element" class = "panel selection_only" >
< p class = "tl" > panel.element.position< / p >
< div class = "toolbar_wrapper element_position" > < / div >
< p class = "tl" > panel.element.size< / p >
< div class = "toolbar_wrapper element_size" > < / div >
< p class = "tl" > panel.element.origin< / p >
< div class = "toolbar_wrapper element_origin" > < / div >
< p class = "tl" > panel.element.rotation< / p >
< div class = "toolbar_wrapper element_rotation" > < / div >
< / div >
2019-04-07 18:53:33 +02:00
< div id = "color" class = "panel" >
< div id = "main_colorpicker_preview" > < div > < / div > < / div >
< input id = "main_colorpicker" >
< / div >
2019-07-19 17:31:22 +02:00
2019-01-09 15:54:35 +01:00
< div id = "outliner" class = "panel grow" >
< div class = "toolbar_wrapper outliner" > < / div >
< ul id = "cubes_list" class = "list" >
< vue-tree :option = "option" > < / vue-tree >
< / ul >
< / div >
2019-07-19 17:31:22 +02:00
2019-04-12 18:44:18 +02:00
< div id = "chat" class = "panel grow" >
< div class = "bar next_to_title" id = "chat_title_bar" > < / div >
< ul id = "chat_history" v-if = "expanded" >
< li v-for = "msg in history" >
< b v-if = "msg.showAuthor()" v-bind:class = "{self: msg.self}" > {{ msg.author }}:< / b >
2019-07-17 18:02:07 +02:00
< span class = "text" v-bind:style = "{color: msg.hex || 'inherit'}" v-html = "msg.html" > < / span >
2019-04-12 18:44:18 +02:00
< span class = "timestamp" > {{ msg.timestamp }}< / span >
< / li >
< / ul >
< div id = "chat_bar" >
< input type = "text" id = "chat_input" class = "dark_bordered f_left" maxlength = "512" >
< i class = "material-icons" onclick = "Chat.send()" > send< / i >
< / div >
< / div >
2019-01-09 15:54:35 +01:00
< / div >
2019-07-17 18:02:07 +02:00
< div id = "center" >
< div id = "preview" >
< / div >
< div id = "timeline" >
< div class = "toolbar_wrapper timeline" > < / div >
2019-08-17 18:26:14 +02:00
< div id = "timeline_vue" >
< div id = "timeline_header" >
< div id = "timeline_corner" v-bind:style = "{width: head_width+'px'}" > < / div >
< div id = "timeline_time_wrapper" >
< div id = "timeline_time" v-bind:style = "{width: (size*length)+'px', left: -scroll_left+'px'}" >
< div v-for = "t in timecodes" class = "timeline_timecode" v-bind:style = "{left: (t.time * size) + 'px'}" >
{{ t.text }}
< / div >
< div id = "timeline_marker"
v-bind:style="{left: (marker * size) + 'px'}"
>< / div >
< / div >
2019-07-17 18:02:07 +02:00
< / div >
< / div >
2019-08-17 18:26:14 +02:00
< div id = "timeline_body" >
< div id = "timeline_body_inner" >
< div id = "timeline_marker_line"
v-bind:style="{left: (8 + marker * size + head_width) + 'px'}"
>< / div >
< li v-for = "animator in animators" class = "animator" :class = "{selected: animator.selected}" :uuid = "animator.uuid" v-on:click = "animator.select();" >
< div class = "animator_head_bar" v-bind:style = "{width: (size*length + head_width)+'px'}" >
< div class = "channel_head" v-bind:style = "{left: scroll_left+'px', width: head_width+'px'}" >
< div class = "text_button" v-on:click . stop = "toggleAnimator(animator)" >
< i class = "icon-open-state fa" v-bind:class = "{'fa-caret-right': !animator.expanded, 'fa-caret-down': animator.expanded}" > < / i >
< / div >
< span v-on:click . stop = "animator.select();" > {{animator.name}}< / span >
< div class = "text_button" v-on:click . stop = "removeAnimator(animator)" >
< i class = "material-icons" > remove< / i >
< / div >
< / div >
< div class = "keyframe_section" v-if = "!animator.expanded" >
< keyframe
v-for="keyframe in animator.keyframes"
v-bind:style="{left: (8 + keyframe.time * size) + 'px'}"
class="keyframe"
v-bind:id="'_'+keyframe.uuid"
>
< i class = "material-icons" > lens< / i >
< / keyframe >
< / div >
< / div >
< div class = "animator_channel_bar" v-bind:style = "{width: (size*length + head_width)+'px'}" v-for = "channel in animator.channels" v-if = "animator.expanded && (!focus_channel || channel == focus_channel)" >
< div class = "channel_head" v-bind:style = "{left: scroll_left+'px', width: head_width+'px'}" >
< span > {{ tl('timeline.'+channel) }}< / span >
< div class = "text_button" v-on:click . stop = "animator.createKeyframe(null, Timeline.time, channel, true)" >
< i class = "material-icons" > add< / i >
< / div >
< / div >
< div class = "keyframe_section" >
< keyframe
v-for="keyframe in animator[channel]"
v-bind:style="{left: (8 + keyframe.time * size) + 'px'}"
class="keyframe"
v-bind:class="[keyframe.channel, keyframe.selected?'selected':'']"
v-bind:id="keyframe.uuid"
v-on:click.stop="keyframe.select($event)"
v-on:dblclick="keyframe.callMarker()"
:title="tl('timeline.'+keyframe.channel)"
@contextmenu.prevent="keyframe.showContextMenu($event)"
>
< i class = "material-icons" > stop< / i >
< / keyframe >
< / div >
< / div >
< / li >
< div id = "timeline_empty_head" class = "channel_head" v-bind:style = "{left: scroll_left+'px', width: head_width+'px'}" >
< / div >
< div id = "timeline_selector" class = "selection_rectangle" > < / div >
< / div >
2019-07-17 18:02:07 +02:00
< / div >
2019-01-09 15:54:35 +01:00
< / div >
2019-07-17 18:02:07 +02:00
< / div >
< div id = "start_screen" >
< content >
< section id = "start-files" >
< left >
< h2 class = "tl" > mode.start.new< / h2 >
< ul >
2019-07-26 13:33:29 +02:00
< li v-for = "format in formats" v-if = "format.show_on_start_screen" v-on:click = "format.new()" >
2019-07-17 18:02:07 +02:00
< span class = "icon_wrapper f_left" v-html = "Blockbench.getIconNode(format.icon).outerHTML" > < / span >
< h3 > {{ format.name }}< / h3 >
< p > {{ format.description }}< / p >
< / li >
< / ul >
< / left >
< right >
< h2 class = "tl" > mode.start.recent< / h2 >
< ul >
< li v-on:click = "openProject(project, $event)" v-for = "project in recent" v-bind:title = "project.path" >
< span class = "icon_wrapper" v-html = "Blockbench.getIconNode(project.icon).outerHTML" > < / span >
< h3 > {{ project.name }}< / h3 >
< p class = "f_right" > {{ getDate(project) }}< / p >
< / li >
< div v-if = "recent.length == 0" > {{ tl('mode.start.no_recents') }}< / div >
< / ul >
< button class = "tl" style = "margin-top: 20px;" onclick = "BarItems.open_model.trigger()" > action.open_model< / button >
< / right >
< / section >
< / content >
2019-01-09 15:54:35 +01:00
< / div >
< / div >
2019-07-17 18:02:07 +02:00
< div id = "status_bar" @ contextmenu = "Interface.status_bar.menu.show(event)" >
2019-01-09 15:54:35 +01:00
< div id = "status_saved" >
< i class = "material-icons" v-if = "Prop.project_saved" v-bind:title = "tl('status_bar.saved')" > check< / i >
< i class = "material-icons" v-else v-bind:title = "tl('status_bar.unsaved')" > close< / i >
< / div >
2019-08-17 18:26:14 +02:00
< div v-html = "Blockbench.getIconNode(Format.icon).outerHTML" v-bind:title = "Format.name" > < / div >
2019-01-09 15:54:35 +01:00
< div id = "status_name" >
{{ Prop.file_name }}
< / div >
< div id = "status_message" class = "hidden" > < / div >
< div class = "f_right" >
{{ Prop.zoom }}%
< / div >
< div class = "f_right" >
{{ Prop.fps }} FPS
< / div >
2019-04-07 18:53:33 +02:00
< div class = "f_right" v-if = "Prop.session" >
{{ Prop.connections }} Clients
< / div >
2019-01-09 15:54:35 +01:00
< div id = "status_progress" v-if = "Prop.progress" v-bind:style = "{width: Prop.progress*100+'%'}" > < / div >
< / div >
2019-07-17 18:02:07 +02:00
< / div >
2019-01-09 15:54:35 +01:00
< script >
initCanvas()
colorSettingsSetup()
animate()
initializeApp()
< / script >
< script >
if (!Blockbench || !Blockbench.setup_successful) {
document.getElementById('loading_error_message').style.display = 'block'
if (typeof require !== undefined) {
require('electron').remote.getCurrentWindow().webContents.openDevTools()
}
2019-07-17 18:02:07 +02:00
} else {
document.getElementById('loading_error_message').innerHTML = 'No loading errors...'
2019-01-09 15:54:35 +01:00
}
< / script >
2017-10-31 14:49:01 +01:00
< / body >
< / html >