From 46e5f39312c730f6666f6ccde70700928e82aceb Mon Sep 17 00:00:00 2001 From: JannisX11 <jannis4236@gmail.com> Date: Sun, 7 Apr 2019 18:53:33 +0200 Subject: [PATCH] v2.6.0 --- css/spectrum.css | 48 +- css/style.css | 1613 ++++++++++++++++++++------------------- index.html | 95 ++- index.php | 98 ++- js/TransformControls.js | 7 +- js/actions.js | 225 +++--- js/animations.js | 139 +++- js/api.js | 211 +++-- js/app.js | 94 ++- js/blockbench.js | 66 +- js/display.js | 4 +- js/edit_sessions.js | 239 ++++++ js/element.js | 258 +++++-- js/interface.js | 85 ++- js/io.js | 292 +++++-- js/keyboard.js | 2 +- js/language.js | 6 +- js/molang.js | 2 +- js/painter.js | 135 ++-- js/plugin_loader.js | 11 +- js/preview.js | 53 +- js/settings.js | 19 +- js/textures.js | 378 ++++----- js/transform.js | 19 +- js/tree.vue.js | 5 +- js/undo.js | 199 +++-- js/util.js | 3 + js/uv.js | 315 +++++--- js/web.js | 3 + lang/de.json | 55 +- lang/en.json | 56 +- lang/es.json | 65 +- lang/fr.json | 125 ++- lang/ja.json | 53 +- lang/nl.json | 67 +- lang/pl.json | 55 +- lang/pt.json | 903 ++++++++++++++++++++++ lang/ru.json | 59 +- lang/sv.json | 53 +- lang/zh.json | 64 +- lib/peer.min.js | 1 + lib/spectrum.js | 3 +- main.js | 9 +- package.json | 2 +- 44 files changed, 4325 insertions(+), 1869 deletions(-) create mode 100644 js/edit_sessions.js create mode 100644 lang/pt.json create mode 100644 lib/peer.min.js diff --git a/css/spectrum.css b/css/spectrum.css index 9e9e8662..08f2c983 100644 --- a/css/spectrum.css +++ b/css/spectrum.css @@ -13,11 +13,32 @@ License: MIT *display: inline; *zoom: 1; /* https://github.com/bgrins/spectrum/issues/40 */ - z-index: 9999994; + z-index: 1; overflow: hidden; } +.sp-container:not(.sp-flat) { + z-index: 22; +} .sp-container.sp-flat { position: relative; + background: transparent; + box-shadow: none; + border: none; + width: 100%; +} +.sp-container.sp-flat { + position: relative; + background: transparent; + box-shadow: none; +} +.sp-container.sp-flat .sp-picker-container { + width: calc(100% - 70px); +} +.sp-container.sp-flat .sp-button-container { + display: none; +} +.sp-container.sp-flat .sp-input-container { + width: 100%; } /* Fix for * { box-sizing: border-box; } */ @@ -46,19 +67,19 @@ License: MIT top:0; left:0; bottom:0; - right:20%; + right: 32px; } .sp-hue { position: absolute; top:0; right:0; bottom:0; - left:84%; + width: 24px; height: 100%; } .sp-clear-enabled .sp-hue { - top:33px; + top:30px; height: 77.5%; } @@ -104,9 +125,6 @@ License: MIT height: 22px; margin-top: 16px; } -.sp-alpha-inner { - border: solid 1px var(--color-border); -} .sp-clear { display: none; @@ -288,9 +306,6 @@ See http://bgrins.github.io/spectrum/themes/ for instructions. .sp-top { margin-bottom: 3px; } -.sp-color, .sp-hue, .sp-clear { - border: solid 1px var(--color-border); -} /* Input */ .sp-input-container { @@ -393,7 +408,7 @@ See http://bgrins.github.io/spectrum/themes/ for instructions. outline: none; } .sp-replacer:hover, .sp-replacer.sp-active { - color: var(--color-text_acc); + color: var(--color-light); } .sp-replacer.sp-disabled { cursor:default; @@ -411,7 +426,6 @@ See http://bgrins.github.io/spectrum/themes/ for instructions. position:relative; width:25px; height: 20px; - border: solid 1px #222; margin-right: 5px; float:left; z-index: 0; @@ -450,6 +464,16 @@ See http://bgrins.github.io/spectrum/themes/ for instructions. text-decoration:none; } +#main_colorpicker_preview { + width: 100%; + height: 12px; + background-image: url(); +} +#main_colorpicker_preview > div { + width: 100%; + height: 12px; +} + .sp-palette span:hover, .sp-palette span.sp-thumb-active { border-color: #000; diff --git a/css/style.css b/css/style.css index 640a0b8c..daf27fe5 100644 --- a/css/style.css +++ b/css/style.css @@ -7,12 +7,14 @@ outline: none; outline-color: rgba(0, 0, 0, 0); } + /*Webkit*/ a:-webkit-any-link { color: inherit; text-decoration: underline; } - - + a:-webkit-any-link:hover { + color: var(--color-light); + } ::-webkit-scrollbar { width: 6px; height: 6px; @@ -21,6 +23,9 @@ ::-webkit-scrollbar-track { background: var(--color-ui); } + ::-webkit-scrollbar-corner { + background: var(--color-ui); + } ::-webkit-scrollbar-thumb { background: var(--color-selected); @@ -29,6 +34,12 @@ ::-webkit-scrollbar-thumb:hover { background: var(--color-accent); } + /*Mozilla*/ + * { + scrollbar-width: thin; + scrollbar-color: var(--color-selected) var(--color-back); + } + ::selection { background: var(--color-accent); border-radius: 4px; @@ -50,18 +61,18 @@ local('Montserrat'), url(../font/Montserrat-Regular.ttf) format('truetype'); } + @font-face { + font-family: 'icomoon'; + src: url('../font/icomoon.eot?t564px'); + src: url('../font/icomoon.eot?t564px#iefix') format('embedded-opentype'), + url('../font/icomoon.ttf?t564px') format('truetype'), + url('../font/icomoon.woff?t564px') format('woff'), + url('../font/icomoon.svg?t564px#icomoon') format('svg'); + font-weight: normal; + font-style: normal; + } -@font-face { - font-family: 'icomoon'; - src: url('../font/icomoon.eot?t564px'); - src: url('../font/icomoon.eot?t564px#iefix') format('embedded-opentype'), - url('../font/icomoon.ttf?t564px') format('truetype'), - url('../font/icomoon.woff?t564px') format('woff'), - url('../font/icomoon.svg?t564px#icomoon') format('svg'); - font-weight: normal; - font-style: normal; -} - +/*Icons*/ [class^="icon-"]:not(.fa), [class*=" icon-"]:not(.fa) { /* use !important to prevent issues with browser extensions that change fonts */ font-family: 'icomoon' !important; @@ -80,119 +91,115 @@ } .icon-blockbench_file:before { - content: "\e900"; + content: "\e900"; } .icon-vertexsnap:before { - content: "\e901"; + content: "\e901"; } .icon-create_bitmap:before { - content: "\e902"; + content: "\e902"; } .icon-objects:before { - content: "\e903"; + content: "\e903"; } .icon-bow:before { - content: "\e904"; + content: "\e904"; } .icon-bb_interface:before { - content: "\e905"; + content: "\e905"; } .icon-blockbench:before { - content: "\e906"; + content: "\e906"; } .icon-x11:before { - content: "\e907"; + content: "\e907"; } .icon-baby_zombie:before { - content: "\e908"; + content: "\e908"; } .icon-armor_stand:before { - content: "\e909"; + content: "\e909"; } .icon-armor_stand_small:before { - content: "\e90a"; + content: "\e90a"; } .icon-ground:before { - content: "\e90b"; + content: "\e90b"; } .icon-hud:before { - content: "\e90c"; + content: "\e90c"; } .icon-inventory_full:before { - content: "\e90d"; + content: "\e90d"; } .icon-inventory_nine:before { - content: "\e90e"; + content: "\e90e"; } .icon-inventory_single:before { - content: "\e90f"; + content: "\e90f"; } .icon-player_head:before { - content: "\e910"; + content: "\e910"; } .icon-zombie:before { - content: "\e911"; + content: "\e911"; } .icon-blockbench_inverted:before { - content: "\e912"; + content: "\e912"; } .icon-optifine_file:before { - content: "\e913"; + content: "\e913"; } .icon-saved:before { - content: "\e914"; + content: "\e914"; } .icon-player:before { - content: "\e915"; + content: "\e915"; } .icon-mirror_x:before { - content: "\e916"; + content: "\e916"; } .icon-mirror_y:before { - content: "\e917"; + content: "\e917"; } .icon-mirror_z:before { - content: "\e918"; + content: "\e918"; } - - - - - .material-icons { - font-family: 'Material Icons'; - font-weight: normal; - font-style: normal; - font-size: 24px; - display: inline-block; - line-height: 1; - text-transform: none; - letter-spacing: normal; - word-wrap: normal; - white-space: nowrap; - direction: ltr; - - -webkit-font-smoothing: antialiased; - text-rendering: optimizeLegibility; - } - i.fa_big { - font-size: 16pt; - height: 30px; - width: 24px; - text-align: center; - vertical-align: text-top; - } - .dialog .message_box_icon { - font-size: 40pt; - float: left; - padding-right: 8px; - min-width: 61px; - max-height: 54px; - max-width: 62px; - } - .dialog img.message_box_icon { - min-width: auto; - } + .material-icons { + font-family: 'Material Icons'; + font-weight: normal; + font-style: normal; + font-size: 24px; + max-width: 24px; + display: inline-block; + line-height: 1; + text-transform: none; + letter-spacing: normal; + word-wrap: normal; + white-space: nowrap; + direction: ltr; + -webkit-font-smoothing: antialiased; + text-rendering: optimizeLegibility; + } + i.fa_big { + font-size: 16pt; + height: 30px; + width: 24px; + text-align: center; + vertical-align: text-top; + } + .dialog .message_box_icon { + font-size: 40pt; + float: left; + padding-right: 8px; + min-width: 61px; + max-height: 54px; + max-width: 62px; + } + .dialog img.message_box_icon { + min-width: auto; + } /*Vars*/ body { @@ -215,15 +222,13 @@ height: 100%; overflow-y: hidden; } - - body { height: 100%; width: 100%; position: fixed; font-family: segoe ui, Helvetiva Neue, arial, sans-serif; - font-size: 18px; + font-size: 17px; font-weight: lighter; color: var(--color-text); @@ -231,6 +236,9 @@ background-color: var(--color-dark); image-rendering: pixelated; } + div { + cursor: default; + } a { text-decoration: none; cursor: default; @@ -243,90 +251,32 @@ b { font-weight: bolder; } - + li { + list-style: none; + cursor: default; + } + hr { + border-top: 1px solid var(--color-dark); + } h2 { font-weight: lighter; margin: 0; font-family: inherit; } h3 { + display: inline-block; font-size: 1.26em; padding-top: 6px; padding-bottom: 4px; font-family: inherit; + font-weight: inherit; + margin-left: 16px; + min-width: 10px; + height: 32px; } h4 { font-family: inherit; } - li { - list-style: none; - cursor: default; - } - div { - cursor: default; - } - div.tool.wide { - width: 79px; - padding-top: 0px; - } - .hidden, .tooltip_shift, .custom_select ul, .mobile_only, .m_disp { - display: none; - } - div.selection_only { - visibility: hidden; - } - a.open-in-browser { - cursor: pointer; - } - .f_left { - float: left; - } - .f_right { - float: right !important; - } - .i_b { - display: inline-block; - } - label.inline_label { - padding-left: 8px; - padding-right: 8px; - padding-top: 2px; - } - .progress_bar { - background-color: var(--color-back); - height: 20px; - width: 488px; - margin-left: 12px; - margin-bottom: 24px; - margin-top: 12px; - } - .progress_bar_inner { - background-color: var(--color-accent); - height: 100%; - width: 0px; - } - .accent_color { - color: var(--color-accent); - font-weight: normal; - } - .slash { - color: var(--color-light); - padding-left: 3px; - padding-right: 3px; - font-weight: normal; - display: inline-block; - } - -/*Axis Colors*/ - .color_x { - color: #d50a0a; - } - .color_y { - color: #23d400; - } - .color_z { - color: #0894ed; - } /*Inputs*/ input { @@ -350,6 +300,13 @@ vertical-align: middle; cursor: default; outline: none; + height: 38px; + min-width: 100px; + width: auto; + color: var(--color-text); + padding-right: 16px; + padding-left: 16px; + font-weight: normal; } button:hover { background: var(--color-accent); @@ -361,7 +318,15 @@ outline: none; outline-color: var(--color-accent); border: none; - background-color: var(--color-back); + background-color: var(--color-button); + height: 32px; + padding-top: 2px; + padding-left: 10px; + padding-right: 10px; + color: var(--color-text); + } + select:hover{ + color: var(--color-light); } select option { -webkit-appearance: none; @@ -381,10 +346,6 @@ resize: none; outline: none; } - .code { - font-family: consolas, monospace; - font-size: 16px; - } input[type="number"]::-webkit-outer-spin-button, input[type="number"]::-webkit-inner-spin-button, { -webkit-appeareance: none; } @@ -486,6 +447,297 @@ background-color: var(--color-selected); } +/*Defaults*/ + div.tool.wide { + width: 79px; + padding-top: 0px; + } + .hidden, .tooltip_shift, .custom_select ul, .mobile_only, .m_disp { + display: none; + } + div.selection_only { + visibility: hidden; + } + a.open-in-browser { + cursor: pointer; + } + .f_left { + float: left; + } + .f_right { + float: right !important; + } + .i_b { + display: inline-block; + } + label.inline_label { + padding-left: 8px; + padding-right: 8px; + padding-top: 2px; + } + .progress_bar { + background-color: var(--color-back); + height: 20px; + margin-top: 12px; + } + .progress_bar_inner { + background-color: var(--color-accent); + height: 100%; + width: 0px; + } + .accent_color { + color: var(--color-accent); + font-weight: normal; + } + .slash { + color: var(--color-light); + padding-left: 3px; + padding-right: 3px; + font-weight: normal; + display: inline-block; + } + .code { + font-family: consolas, monospace; + font-size: 16px; + } + .small_text { + font-size: 0.94em; + } + .color_x { + color: #d50a0a; + } + .color_y { + color: #23d400; + } + .color_z { + color: #0894ed; + } + .button { + display: inline-block; + width: 30px; + text-align: center; + cursor: default; + } + .dark_bordered { + height: 32px; + padding-left: 4px; + background-color: var(--color-back); + border: 1px solid var(--color-border); + } + .input_wide { + width: 100%; + padding: 8px; + height: 40px; + padding-bottom: 5px; + } + +/*General*/ + canvas.preview { + background-repeat: no-repeat; + } + .text_padding { + margin-left: 5px; + margin-right: 5px; + } + .toolbar_label { + margin-top: 4px; + } + .bar { + height: 32px; + margin-top: 2px; + } + .bar > * { + float: left; + } + .list { + background-color: var(--color-back); + height: calc(100% - 86px); + width: calc(100% - 2px); + overflow-y: scroll; + flex-grow: 1; + } + body.animation_mode #cubes_list { + height: calc(100% - 50px); + } + body.animation_mode #cubes_list .outliner_object.cube { + display: none; + } + #quick_message_box { + position: fixed; + left: 0; + right: 0; + margin-left: auto; + margin-right: auto; + margin-top: 420px; + z-index: 100; + min-width: 100px; + max-width: 200px; + background-color: var(--color-bright_ui); + color: var(--color-text_acc); + box-shadow: 0 0 2px black; + text-align: center; + cursor: default; + pointer-events: none; + } + .uv_message_box { + position: absolute; + margin-left: auto; + margin-right: auto; + z-index: 101; + min-width: 100px; + max-width: 200px; + background-color: var(--color-ui); + color: var(--color-light); + box-shadow: 0 0 2px black; + text-align: center; + cursor: default; + top: 40px; + left: 60px; + } + #selection_box { + position: absolute; + display: block; + border: 1px solid var(--color-accent); + background-color: rgba(40,50,60,0.5); + pointer-events: none; + } + + .spinning { + -webkit-animation: spin 2s linear infinite; + -moz-animation: spin 2s linear infinite; + animation: spin 2s linear infinite; + } + @-moz-keyframes spin { 100% { -moz-transform: rotate(360deg); } } + @-webkit-keyframes spin { 100% { -webkit-transform: rotate(360deg); } } + @keyframes spin { 100% { -webkit-transform: rotate(360deg); transform:rotate(360deg); } } + +/*Dialog*/ + #blackout { + display: none; + position: absolute; + top: 0; + left: 0; + right: 0; + bottom: 0; + width: 100%; + height: 100%; + background-color: #000; + opacity: 0.5; + z-index: 17; + } + #dialog_close_button { + position: absolute; + z-index: 102; + right: 5px; + top: 5px; + height: 28px; + } + #dialog_close_button:hover { + color: var(--color-light); + } + .dialog:not(.draggable) #dialog_close_button { + top: 8px; + right: -34px; + } + + .dialog { + display: none; + position: fixed; + width: 540px; + top: 0; + background-color: var(--color-ui); + border: 1px solid var(--color-border); + z-index: 18; + box-shadow: 0 0px 40px #000; + } + .dialog:not(.draggable) { + left: 0; + right: 0; + top:0; + margin-right: auto; + margin-left: auto; + } + .dialog_bar { + min-height: 32px; + margin-top: 4px; + margin-bottom: 4px; + height: auto; + } + .dialog_bar.narrow { + min-height: 30px; + } + .dialog_bar.button_bar { + text-align: right; + } + .dialog_bar button.large { + margin-bottom: 0; + margin-top: 16px; + } + .dialog .tab { + float: left; + width: 25%; + height: 34px; + text-align: center; + padding-top: 4px; + background: var(--color-dark); + vertical-align: middle; + cursor: default; + } + .dialog#settings .tab:firstChild { + margin-left: 0; + width: 25%; + } + .dialog .tab:hover { + color: var(--color-light); + } + .dialog .tab.open { + background: transparent; + border-top: none; + transition: border-top 200 ease-in; + } + .dialog .tab_content { + height: calc(100% - 90px); + width: 100%; + padding-bottom: 16px; + } + #settings_tab_bar { + margin: -24px; + margin-bottom: 0; + } + .dialog p { + margin-top: 8px; + margin-bottom: 8px; + } + .dialog h3 { + margin-left: 16px; + } + .dialog.draggable h2 { + padding-left: 26px; + padding-top: 2px; + margin-top: -24px; + margin-left: -24px; + margin-right: -24px; + background: var(--color-button); + height: 32px; + margin-bottom: 20px; + font-size: 1.12em; + } + .dialog_handle { + cursor: pointer; + overflow: hidden; + } + .dialog_bar label.in_toolbar { + padding-left: 0; + } + .dialog p.multiline_text { + margin-top: 0; + margin-bottom: 20px; + margin-left: 12px; + margin-right: 12px; + font-size: 0.86em; + user-select: text; + } + /*Menu Bar*/ ul#menu_bar { height: 32px; @@ -496,6 +748,7 @@ padding-top: 2px; padding-bottom: 0; float: left; + height: 100%; } li.menu_bar_point.opened { color: var(--color-light); @@ -506,8 +759,8 @@ } header div#title { width: auto; - padding-right: 16px; - padding-left: 10px; + padding-right: 8px; + padding-left: 8px; font-size: 1.2em; font-weight: normal; font-family: montserrat, arial, sans-serif; @@ -515,6 +768,7 @@ color: var(--color-light); vertical-align: top; min-width: 62px; + padding-top: 2px; } header div#title i { display: none; @@ -576,11 +830,123 @@ width: 16px; margin-left: -5px; } + .tool { + height: 32px; + width: 40px; + margin-left: 1px; + margin-right: 1px; + background: transparent; + display: inline-block; + text-align: center; + vertical-align: middle; + cursor: default; + float: left; + color: var(--color-text); + } + .tool i { + display: inline-block; + margin-top: 4px; + } + + .tool.sel { + border-bottom: 4px solid var(--color-accent); + } + + .tool.widget { + width: auto; + padding: 0; + } + .tool.widget.bar_text { + padding: 3px; + } + + .tool:hover { + color: var(--color-light); + } + .tool.head_right { + margin-top: -29px; + height: 24px; + float: right; + } + .tool.right_tool { + position: relative; + } + .toolbar_wrapper.tools > .toolbar { + margin-left: 20px; + margin-right: 20px; + } + + .placeholder { + width: 20px; + height: 10px; + float: left; + } + .toolbar_separator { + width: 2px; + height: 24px; + float: left; + background-color: var(--color-border); + margin: 4px; + margin-bottom: 0; + } + .text_button:hover { + color: var(--color-light); + } + select.tool { + -webkit-appearance: none; + width: 112px; + padding-left: 5px; + background-color: var(--color-back); + border: 1px solid var(--color-border); + } + .bar_select { + position: relative; + } + .bar_select:hover { + color: var(--color-light); + } + .bar_select select { + padding-right: 24px; + } + .bar_select::before { + content: "\23F7"; + display: block; + position: absolute; + height: 12px; + width: 16px; + pointer-events: none; + right: 6px; + top: 3px; + } + .half { + width: calc(50% - 4px); + } + .tooltip { + position: absolute; + height: 30px; + padding-left: 5px; + padding-right: 5px; + padding-top: 2px; + color: var(--color-text_acc); + margin-top: 32px; + display: none; + background: var(--color-bright_ui); + white-space: nowrap; + z-index: 125; + box-shadow: 0 0.4px 3.5px rgba(0, 0, 0, 0.6); + } + .tool:hover > .tooltip:not(:hover) { + display: block; + } + .tooltip_shift { + display: inline; + display: none; + } /*Layout*/ body { display: grid; - grid-template-columns: 328px auto 300px; + grid-template-columns: 338px auto 300px; grid-template-rows: 32px calc(100% - 58px) 26px; grid-template-areas: "header header header" @@ -692,179 +1058,6 @@ pointer-events: none; } -/*General*/ - hr { - border-top: 1px solid var(--color-dark); - } - .input_wide { - width: 100%; - padding: 5px; - background-color: var(--color-back); - } - canvas.preview { - background-repeat: no-repeat; - } - .bar { - height: 32px; - margin-top: 2px; - } - .bar > * { - float: left; - } - .text_padding { - margin-left: 5px; - margin-right: 5px; - } - .toolbar_label { - margin-top: 4px; - } - - .tool { - height: 32px; - width: 40px; - margin-left: 1px; - margin-right: 1px; - background: transparent; - display: inline-block; - text-align: center; - vertical-align: middle; - cursor: default; - float: left; - color: var(--color-text); - } - .tool i { - display: inline-block; - margin-top: 4px; - } - - .tool.sel { - border-bottom: 4px solid var(--color-accent); - } - - .tool.widget { - width: auto; - padding: 0; - } - .tool.widget.bar_text { - padding: 3px; - } - - .tool:hover { - color: var(--color-light); - } - .tool.head_right { - margin-top: -29px; - height: 24px; - float: right; - } - .tool.right_tool { - position: relative; - } - .toolbar_wrapper.tools > .toolbar { - margin-left: 20px; - margin-right: 20px; - } - - .placeholder { - width: 20px; - height: 10px; - float: left; - } - .toolbar_separator { - width: 2px; - height: 24px; - float: left; - background-color: var(--color-border); - margin: 4px; - margin-bottom: 0; - } - .text_button:hover { - color: var(--color-light); - } - - select.tool { - -webkit-appearance: none; - width: 112px; - padding-left: 5px; - background-color: var(--color-back); - border: 1px solid var(--color-border); - } - - .half { - width: calc(50% - 4px); - } - - .tooltip { - position: absolute; - height: 30px; - padding-left: 5px; - padding-right: 5px; - padding-top: 2px; - color: var(--color-text_acc); - margin-top: 32px; - display: none; - background: var(--color-bright_ui); - white-space: nowrap; - z-index: 125; - box-shadow: 0 0.4px 3.5px rgba(0, 0, 0, 0.6); - } - - .tool:hover > .tooltip:not(:hover) { - display: block; - } - .tooltip_shift { - display: inline; - display: none; - } - - .button { - display: inline-block; - width: 30px; - text-align: center; - cursor: default; - } - - button.large { - padding: 8px; - margin-bottom: 8px; - margin-left: 8px; - height: 40px; - min-width: 160px; - width: auto; - color: var(--color-text); - } - - .list { - background-color: var(--color-back); - height: calc(100% - 86px); - width: calc(100% - 2px); - overflow-y: scroll; - flex-grow: 1; - } - body.animation_mode #cubes_list { - height: calc(100% - 50px); - } - body.animation_mode #cubes_list .outliner_object.cube { - display: none; - } - - h3 { - font-weight: inherit; - margin-left: 16px; - min-width: 10px; - height: 32px; - display: inline-block; - } - - .spinning { - -webkit-animation: spin 2s linear infinite; - -moz-animation: spin 2s linear infinite; - animation: spin 2s linear infinite; - } - @-moz-keyframes spin { 100% { -moz-transform: rotate(360deg); } } - @-webkit-keyframes spin { 100% { -webkit-transform: rotate(360deg); } } - @keyframes spin { 100% { -webkit-transform: rotate(360deg); transform:rotate(360deg); } } - /*Displaytabs*/ .tabs_small input[type="radio"]:checked+label { @@ -906,6 +1099,9 @@ #cubes_list > div.vue-tree > ul > li > *:not(ul) { display: none; } + #cubes_list > div > ul > li > ul > li:last-child { + margin-bottom: 180px; + } .outliner_node .outliner_object i, .outliner_object i[class^="icon-"] { text-align: center; width: 18px; @@ -992,6 +1188,30 @@ font-size: 13pt; float: right; } + .outliner_object i.ec_0 { + color: #A2EBFF; /*light_blue*/ + } + .outliner_object i.ec_1 { + color: #FFF899; /*yellow*/ + } + .outliner_object i.ec_2 { + color: #E8BD7B; /*orange*/ + } + .outliner_object i.ec_3 { + color: #FFA7A4; /*red*/ + } + .outliner_object i.ec_4 { + color: #C5A6E8; /*purple*/ + } + .outliner_object i.ec_5 { + color: #A6C8FF; /*blue*/ + } + .outliner_object i.ec_6 { + color: #7BFFA3; /*green*/ + } + .outliner_object i.ec_7 { + color: #BDFFA6; /*lime*/ + } body > .outliner_object { width: 180px; padding-left: 0 !important; @@ -1032,11 +1252,11 @@ left: 20px; border: 1px solid var(--color-border); } + #options .bar .nslide, #options .bar .tool.wide { + width: 83px; + } - - - -/*Context*/ +/*(Context-)Menu*/ .contextMenu { position: absolute; display: block; @@ -1106,9 +1326,10 @@ background-color: var(--color-text); } -/*Action Select*/ +/*Action Control*/ #action_selector { position: absolute; + z-index: 24; right: 0; left: 0; margin-left: auto; @@ -1131,34 +1352,43 @@ right: 6px; top: 9px; } - #action_selector > ul { + #action_selector > div { + background-color: var(--color-ui); + color: var(--color-text); + width: 340px; + margin-left: 8px; + box-shadow: 0 0 5px black; + } + #action_selector > div > ul { background-color: var(--color-bright_ui); color: var(--color-text_acc); min-height: 20px; - width: 300px; - margin-left: 8px; - box-shadow: 0 0 5px black; + width: 340px; max-height: 400px; overflow-y: auto; overflow-x: hidden; } - #action_selector > ul > li { + #action_selector > div > div { + background-color: var(--color-ui); + color: var(--color-text); + height: auto; + padding: 5px; + font-size: 0.94em + } + #action_selector ul > li { padding: 4px; + overflow: hidden; } - #action_selector > ul > li.selected { + #action_selector ul > li.selected { background-color: var(--color-accent); + color: var(--color-text_acc); } - #action_selector > ul > li div.icon_wrapper { + #action_selector ul > li div.icon_wrapper { display: inline-block; height: 26px; vertical-align: text-top; } - - - - - #bar_item_list { max-height: 400px; margin-bottom: 20px; @@ -1262,7 +1492,6 @@ text-shadow: 0 0 5px #000; } - /*Animations*/ .panel#animations #animations_list { min-height: 126px; @@ -1294,6 +1523,20 @@ .animation > .animation_name { width: calc(100% - 28px); } + .panel#keyframe .tabs_small label { + font-size: 1em; + height: 32px; + width: 25%; + } + .panel#keyframe .bar label { + margin-top: 3px; + margin-left: 8px; + width: 32px; + text-align: center; + } + .panel#keyframe .bar input.dark_bordered { + width: calc(100% - 45px); + } /*Timeline*/ #timeline { @@ -1419,22 +1662,6 @@ margin-top: 28px; border-right: 1px solid var(--color-border); } -/*Keyframe Panel*/ - .panel#keyframe .tabs_small label { - font-size: 1em; - height: 32px; - width: 25%; - } - .panel#keyframe .bar label { - margin-top: 3px; - margin-left: 8px; - width: 32px; - text-align: center; - } - .panel#keyframe .bar input.dark_bordered { - width: calc(100% - 45px); - } - /*UV*/ #uv_dialog { @@ -1475,11 +1702,18 @@ float: right; } + #uv_viewport { + height: 320px; + width: 320px; + background-size: 320px; + position: relative; + overflow: hidden; + } #uv_frame { height: 320px; width: 320px; - margin-left: 3px; - margin-top: 3px; + margin: 4px; + margin-bottom: 0; background-size: 320px; background-repeat: no-repeat; position: relative; @@ -1500,13 +1734,19 @@ #uv_dialog h2.dialog_handle.entity_mode_only { margin: 0; } - #uv_size { height: 320px; width: 320px; cursor: move; box-sizing: border-box; z-index: 1; + box-shadow: 0 0 0 1.5px var(--color-text); + } + #uv_frame:hover #uv_size { + box-shadow: 0 0 0 1.5px white; + } + #uv_frame:hover #uv_size.dark_frame { + box-shadow: 0 0 0 1.5px black; } .uv_size_handle { position: absolute; @@ -1519,11 +1759,12 @@ .uv_transform_info { position: absolute; display: block; - right: 0; - bottom: -3px; + right: 8px; + top: 8px; font-size: 0.8em; cursor: default; pointer-events: none; + z-index: 5; } #uv .bar.next_to_title { margin-top: -32px; @@ -1532,145 +1773,43 @@ } #uv .bar.next_to_title .tool { } - -/*Options*/ - #cube_axis { - width: 58px; - position: relative; - z-index: 5; - } - #options .bar .nslide { - width: 83px; - } - #options .bar .tool.wide { - width: 83px; - } - .tool .tooltip.clip_right { - right: 0; - } - -/*Dialog*/ - #blackout { - display: none; + #uv_size .ui-resizable-se:before, + #uv_size .ui-resizable-sw:before, + #uv_size .ui-resizable-ne:before, + #uv_size .ui-resizable-nw:before { + content: ""; position: absolute; - top: 0; - left: 0; - right: 0; - bottom: 0; - width: 100%; - height: 100%; - background-color: #000; - opacity: 0.5; - z-index: 17; + height: 8px; + width: 8px; + background-color: var(--color-text); } - #dialog_close_button { - position: absolute; - z-index: 102; - right: 5px; - top: 5px; - height: 28px; + #uv_frame:hover #uv_size .ui-resizable-handle:before { + background-color: white; } - #dialog_close_button:hover { - color: var(--color-light); + #uv_frame:hover #uv_size.dark_frame .ui-resizable-handle:before { + background-color: black; } - .dialog:not(.draggable) #dialog_close_button { - top: 8px; - right: -34px; + #uv_frame:hover #uv_size .ui-resizable-handle:hover:before { + background-color: var(--color-accent); + } + #uv_size .ui-resizable-se:before { + bottom: 1px; + right: 1px; + } + #uv_size .ui-resizable-sw:before { + bottom: 1px; + left: 1px; + } + #uv_size .ui-resizable-ne:before { + top: 1px; + right: 1px; + } + #uv_size .ui-resizable-nw:before { + top: 1px; + left: 1px; } - .dialog { - display: none; - position: fixed; - width: 540px; - top: 0; - background-color: var(--color-ui); - border: 1px solid var(--color-border); - z-index: 18; - box-shadow: 0 0px 40px #000; - } - .dialog:not(.draggable) { - left: 0; - right: 0; - top:0; - margin-right: auto; - margin-left: auto; - } - .dialog_bar { - min-height: 50px; - height: auto; - } - .dialog_bar.narrow { - min-height: 30px; - } - .dialog_bar.button_bar { - text-align: right; - } - .dialog .tab { - float: left; - width: calc(25% - 2px); - height: 34px; - text-align: center; - padding-top: 4px; - margin-left: 2px; - background: var(--color-dark); - vertical-align: middle; - cursor: default; - } - .dialog#settings .tab:firstChild { - margin-left: 0; - width: 25%; - } - .dialog .tab:hover { - color: var(--color-light); - } - .dialog .tab.open { - background: transparent; - border-top: none; - transition: border-top 200 ease-in; - } - .dialog .tab_content { - height: calc(100% - 90px); - width: 100%; - padding-bottom: 16px; - } - #settings_tab_bar { - margin: -24px; - margin-bottom: 0; - } - .dialog p { - margin-top: 16px; - } - .dialog h3 { - margin-left: 16px; - } - .dialog.draggable h2 { - padding-left: 26px; - margin-top: -24px; - margin-left: -24px; - margin-right: -24px; - background: var(--color-button); - height: 32px; - margin-bottom: 20px; - font-size: 1.12em; - } - .dialog_handle { - cursor: pointer; - overflow: hidden; - } - .dialog_bar label.in_toolbar { - padding-left: 0; - } - - #keybindlist { - max-height: 600px; - margin-bottom: 20px; - overflow-y: scroll; - } - #keybindlist > li { - width: 100%; - min-height: 34px; - padding-left: 6px; - } +/*Settings Dialog*/ #settings h3 > i { margin-top: 5px; float: left; @@ -1678,42 +1817,9 @@ #settings h3:hover { color: var(--color-light); } - #keybindlist .tool { - height: 32px; - width: 25px; - float: right; - } - #keybindlist li > div.keybindslot { - width: calc(48% - 32px); - padding: 2px; - padding-left: 8px; - height: 30px; - background-color: var(--color-button); - border: 1px solid var(--color-border); - font-size: 0.84em; - overflow: hidden; - white-space: nowrap; - display: inline-block; - } - #keybindlist li > div.keybindslot:hover { - background-color: var(--color-selected); - color: var(--color-light); - } - #keybindlist li > div:first-child { - background: transparent; - width: calc(52% - 28px); - text-align: right; - padding: 2px; - padding-left: 8px; - vertical-align: top; - display: inline-block; - } - #keybindlist > li > div:focus { - color: transparent; - text-shadow: 0 0 0 var(--color-light); - background-color: var(--color-dark); - } + + /*Settings*/ #settings li h3 { margin: 0; padding: 0; @@ -1740,7 +1846,7 @@ #settingslist li > .setting_label { display: inline-block; margin-left: 8px; - width: 420px; + width: 416px; } #settingslist .setting_name { font-size: 1.1em; @@ -1762,10 +1868,13 @@ padding: 10px; margin-left: 5px; } - #settingslist select { + #settingslist div.bar_select { margin: 8px; width: 96%; } + #settingslist div.bar_select select { + width: 100%; + } #settingslist li li i { font-size: 27pt; width: 34px; @@ -1775,11 +1884,97 @@ color: var(--color-light); } + /*Keybinds*/ + #keybindlist .tool { + height: 32px; + width: 25px; + float: right; + } + #keybindlist li > div.keybindslot { + width: calc(48% - 32px); + padding: 2px; + padding-left: 8px; + height: 30px; + background-color: var(--color-button); + border: 1px solid var(--color-border); + font-size: 0.84em; + overflow: hidden; + white-space: nowrap; + display: inline-block; + } + #keybindlist li > div.keybindslot:hover { + background-color: var(--color-selected); + color: var(--color-light); + } + #keybindlist li > div:first-child { + background: transparent; + width: calc(52% - 28px); + text-align: right; + padding: 2px; + padding-left: 8px; + vertical-align: top; + display: inline-block; + } + #keybindlist > li > div:focus { + color: transparent; + text-shadow: 0 0 0 var(--color-light); + background-color: var(--color-dark); + } + #keybindlist { + max-height: 600px; + margin-bottom: 20px; + overflow-y: scroll; + } + #keybindlist > li { + width: 100%; + min-height: 34px; + padding-left: 6px; + } + /*Colors*/ + div#color_wrapper { + columns: 2; + margin-bottom: 20px; + } + .color_field { + min-height: 64px; + width: 100% + } + .color_field .desc { + width: 172px; + display: inline-block; + } + .color_field p { + margin: 0; + font-size: 0.8em; + } + .color_field h4 { + margin: 0; + font-size: 1.2em; + } + input.color_input { + -webkit-appearance: none; + display: none; + } + label.color_input { + -webkit-appearance: none; + appearance: none; + outline: none; + height: 36px; + width: 46px; + margin: 4px; + margin-top: 10px; + display: inline-block; + vertical-align: top; + border: 1px solid var(--color-border); + } + /*About*/ #credits a:hover { text-decoration: underline; } + +/*Specific Dialogs*/ .dialog#updater .dialog_bar.narrow { padding-top: 3px; } @@ -1789,14 +1984,6 @@ margin-top: -64px; width: 240px; } - .dialog p.multiline_text { - margin-top: 0; - margin-bottom: 20px; - margin-left: 12px; - margin-right: 12px; - font-size: 0.86em; - user-select: text; - } .dialog#texture_edit p.multiline_text { width: 344px; min-height: 51px; @@ -1823,8 +2010,103 @@ width: 73px; border: none; } + /*Scale*/ + #model_scale_range { + width: calc(100% - 50px); + float: left; + height: 31px; + padding-top: 3px; + } + #model_scale_label { + width: 50px; + padding-top: 3px; + text-align: center; + float: left; + } + #scaling_clipping_warning { + color: #ff384b; + } + .borderless { + margin: 0; + padding: 0; + border: none; + } -/*Keybind recording*/ + /*Extrusion*/ + #image_extruder label { + float: left; + margin-right: 8px; + padding-top: 5px; + } + #scan_tolerance { + width: 200px; + } + #scan_tolerance_label { + margin-left: 8px; + } + #extrusion_canvas { + margin-left: 12px; + border-bottom: 1px solid #000; + border-right: 1px solid #000; + } + button.large:first-child { + margin-left: 0; + } + + /*Selection Creator*/ + input[type=range].dark_bordered { + height: 32px; + padding-top: 3px; + padding-left: 0; + } + select.dark_bordered { + color: var(--color-text); + padding: 6px; + padding-top: 2px; + height: 32px; + } + label.name_space_left { + float: left; + min-width: 155px; + margin-left: 1px; + margin-right: 8px; + } + label.in_toolbar { + text-align: left; + color: var(--color-text); + padding: 8px; + padding-top: 3px; + padding-bottom: 0; + } + + /*PE Import Dialog*/ + #pe_list { + max-height: 600px; + margin-bottom: 20px; + overflow-y: scroll; + } + #pe_list li > * { + margin: 0; + margin-left: 12px; + cursor: default; + overflow-x: hidden; + } + #pe_list li.selected { + background-color: var(--color-selected); + } + #pe_list li > .pe_icon { + height: 48px; + width: 48px; + float: left; + margin: 4px; + margin-right: 8px; + margin-bottom: 0; + } + #pe_list li > h4 > span { + font-size: 0.7; + } + +/*Keybind Recording*/ #overlay_message_box { height: 100%; width: 100%; @@ -1844,36 +2126,11 @@ } #overlay_message_box h3 i { vertical-align: sub; - padding: 8px; + margin: 8px; font-size: 1.2em; } /*Display*/ - .mode_tab { - display: block; - float: right; - height: 32px; - padding-right: 12px; - padding-left: 12px; - margin-left: 4px; - padding-top: 3px; - font-size: 1.1em; - font-weight: normal; - cursor: pointer; - background-color: var(--color-back); - border: 1px solid var(--color-border); - } - .mode_tab.open { - background-color: var(--color-selected); - border-bottom: none; - cursor: default; - } - .mode_tab:not(.open):hover { - color: var(--color-light); - } - #display_settings { - - } #display_bar .tool, #display_ref_bar > div { width: calc(100% / 8 - 2px); max-width: 52px; @@ -1927,179 +2184,6 @@ margin-right: auto; } - -/*Scale*/ - #model_scale_range { - width: calc(100% - 50px); - float: left; - height: 31px; - padding-top: 3px; - } - #model_scale_label { - width: 50px; - padding-top: 3px; - text-align: center; - float: left; - } - #scaling_clipping_warning { - color: #ff384b; - } - .borderless { - margin: 0; - padding: 0; - border: none; - } - -/*Extrusion*/ - #image_extruder label { - float: left; - margin-right: 8px; - padding-top: 5px; - } - #scan_tolerance { - width: 200px; - } - #scan_tolerance_label { - margin-left: 8px; - } - #extrusion_canvas { - margin-left: 12px; - border-bottom: 1px solid #000; - border-right: 1px solid #000; - } - button.large:first-child { - margin-left: 0; - } - - /*Selection Creator*/ - .dark_bordered { - height: 32px; - padding-left: 4px; - background-color: var(--color-back); - border: 1px solid var(--color-border); - } - input[type=range].dark_bordered { - height: 32px; - padding-top: 3px; - padding-left: 0; - } - select.dark_bordered { - color: var(--color-text); - padding: 6px; - padding-top: 2px; - height: 32px; - } - label.name_space_left { - float: left; - min-width: 155px; - margin-left: 1px; - margin-right: 8px; - } - label.in_toolbar { - text-align: left; - color: var(--color-text); - padding: 8px; - padding-top: 3px; - padding-bottom: 0; - } - -/*Layout Menu*/ - div#color_wrapper { - columns: 2; - margin-bottom: 20px; - } - .color_field { - min-height: 64px; - width: 100% - } - .color_field .desc { - width: 172px; - display: inline-block; - } - .color_field p { - margin: 0; - font-size: 0.8em; - } - .color_field h4 { - margin: 0; - font-size: 1.2em; - } - input.color_input { - -webkit-appearance: none; - display: none; - } - label.color_input { - -webkit-appearance: none; - appearance: none; - outline: none; - height: 36px; - width: 46px; - margin: 4px; - margin-top: 10px; - display: inline-block; - vertical-align: top; - border: 1px solid var(--color-border); - } - - #quick_message_box { - position: fixed; - left: 0; - right: 0; - margin-left: auto; - margin-right: auto; - margin-top: 420px; - z-index: 100; - min-width: 100px; - max-width: 200px; - background-color: var(--color-ui); - color: var(--color-light); - box-shadow: 0 0 2px black; - text-align: center; - cursor: default; - pointer-events: none; - } - .uv_message_box { - position: absolute; - margin-left: auto; - margin-right: auto; - z-index: 101; - min-width: 100px; - max-width: 200px; - background-color: var(--color-ui); - color: var(--color-light); - box-shadow: 0 0 2px black; - text-align: center; - cursor: default; - top: 40px; - left: 60px; - } - -/*PE Import Dialog*/ - #pe_list { - max-height: 600px; - margin-bottom: 20px; - overflow-y: scroll; - } - #pe_list li > * { - margin: 0; - margin-left: 12px; - cursor: default; - overflow-x: hidden; - } - #pe_list li.selected { - background-color: var(--color-selected); - } - #pe_list li > .pe_icon { - height: 48px; - width: 48px; - float: left; - margin: 4px; - margin-right: 8px; - } - #pe_list li > h4 > span { - font-size: 0.7; - } - /*Plugin Menu*/ .bar.next_to_title { display: inline-block; @@ -2233,8 +2317,7 @@ margin-top: 6px; } -/* Toolbar Dialog */ - +/*Toolbar Dialog*/ #bar_item_list { max-height: 400px; margin-bottom: 20px; @@ -2272,9 +2355,7 @@ border: 1px solid var(--color-border); } - /*Status Bar*/ - #status_bar { position: relative; padding: 2px; @@ -2300,33 +2381,3 @@ bottom: 0; left: 0; } - -/*Entity Mode*/ - - a.ml5 i.shade_mirror_on:before { - content: "\f005"; - } - a.ml5 i.shade_mirror_off:before { - content: "\f006"; - } - body.entity_mode a.ml5 i.shade_mirror_on:before { - content: "\f005"; - } - body.entity_mode a.ml5 i.shade_mirror_off:before { - content: "\f005"; - } - - - #cubes_list > div > ul > li > ul > li:last-child { - margin-bottom: 180px; - } - - - -#selection_box { - position: absolute; - display: block; - border: 1px solid var(--color-accent); - background-color: rgba(40,50,60,0.5); - pointer-events: none; -} \ No newline at end of file diff --git a/index.html b/index.html index 6f705665..4788093a 100644 --- a/index.html +++ b/index.html @@ -19,7 +19,7 @@ <script> if (typeof module === 'object') {window.module = module; module = undefined;}//jQuery Fix const isApp = typeof require !== 'undefined'; - const appVersion = '2.5.1'; + const appVersion = '2.6.0'; </script> <script src="lib/vue.min.js"></script> <script src="lib/vue_sortable.js"></script> @@ -30,6 +30,7 @@ <script src="lib/jimp.min.js"></script> <script src="lib/jszip.min.js"></script> <script src="lib/gif.js"></script> + <script src="lib/peer.min.js"></script> <script src="lib/spectrum.js"></script> <script src="lib/three.js"></script> <script src="lib/three_custom.js"></script> @@ -43,6 +44,7 @@ <script src="js/keyboard.js"></script> <script src="js/settings.js"></script> <script src="js/undo.js"></script> + <script src="js/edit_sessions.js"></script> <script type="text/javascript"> if (isApp === true) { @@ -129,7 +131,7 @@ </div> <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> - <button type="button" class="" v-on:click="plugin.download()" v-else><i class="material-icons">add</i><span class="tl">dialog.plugins.install</span></button> + <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> <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> @@ -149,6 +151,35 @@ <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> </div> + <div class="dialog draggable paddinged" id="edit_sessions"> + <h2 class="dialog_handle tl">dialog.edit_session.title</h2> + + <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> + </div> + <div class="dialog draggable paddinged" id="toolbar_edit"> <h2 class="dialog_handle tl">dialog.toolbar_edit.title</h2> @@ -276,9 +307,7 @@ <div class="dialog draggable paddinged" id="scaling"> <h2 class="dialog_handle tl">dialog.scale.title</h2> - <div class="dialog_bar narrow"> - <label class="tl">dialog.scale.axis</label> - </div> + <label class="tl">dialog.scale.axis</label> <div class="dialog_bar" style="height: 32px;"> <input type="checkbox" class="toggle_panel" id="model_scale_x_axis" onchange="scaleAll()" checked> @@ -289,20 +318,28 @@ <label class="toggle_panel" for="model_scale_z_axis">Z</label> </div> - <div class="dialog_bar narrow"> - <label class="tl">dialog.scale.scale</label> + <label class="tl">data.origin</label> + <div class="dialog_bar"> + <label for="scaling_origin_x" class="inline_label tl">X</label> + <input type="number" id="scaling_origin_x" class="dark_bordered mediun_width" oninput="scaleAll()"> + <label for="scaling_origin_y" class="inline_label tl">Y</label> + <input type="number" id="scaling_origin_y" class="dark_bordered mediun_width" oninput="scaleAll()"> + <label for="scaling_origin_z" class="inline_label tl">Z</label> + <input type="number" id="scaling_origin_z" class="dark_bordered mediun_width" oninput="scaleAll()"> </div> + <label class="tl">dialog.scale.scale</label> <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> + <div class="dialog_bar narrow" id="scaling_clipping_warning"></div> <div class="dialog_bar"> - <button type="button" onclick="scaleAll(true)" class="large confirm_btn tl">dialog.scale.confirm</button> - <button type="button" class="large cancel_btn tl" onclick="cancelScaleAll()">dialog.cancel</button> - <button type="button" class="large hidden tl" id="scale_overflow_btn" onclick="scaleAllSelectOverflow()">dialog.scale.select_overflow</button> + <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> + <button type="button" class="hidden tl" id="scale_overflow_btn" onclick="scaleAllSelectOverflow()">dialog.scale.select_overflow</button> </div> <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> </div> @@ -422,7 +459,7 @@ <div class="dialog_bar"> - <button type="button" class="large tl confirm_btn cancel_btn" onclick="saveProjectSettings()">dialog.confirm</button> + <button type="button" class="large tl confirm_btn cancel_btn" onclick="saveProjectSettings();hideDialog();">dialog.confirm</button> <button type="button" class="large tl" id="entity_mode_convert" onclick="entityMode.convert()">dialog.project.to_entitymodel</button> </div> <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> @@ -466,9 +503,11 @@ <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'"> - <select v-model="setting.value" class="dark_bordered"> - <option v-for="(text, id) in setting.options" v-bind:value="id">{{ text }}</option> - </select> + <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> </template> </li> </ul> @@ -697,12 +736,15 @@ <div id="action_selector" v-if="open"> <input type="text" v-model="search_input"> <i class="material-icons" id="action_search_bar_icon">search</i> - <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> - {{ item.name }} - </li> - </ul> + <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> + {{ item.name }} + </li> + </ul> + <div class="small_text" v-if="actions[index]">{{ actions[index].description }}</div> + </div> </div> <header> @@ -864,7 +906,7 @@ > <div class="texture_icon_wrapper"> <img v-bind:texid="texture.id" v-bind:src="texture.source" class="texture_icon" width="48px" alt="[E]" v-if="texture.show_icon" /> - <i class="material-icons texture_error" title="Image Error" v-if="texture.error">error_outline</i> + <i class="material-icons texture_error" v-bind:title="texture.getErrorMessage()" v-if="texture.error">error_outline</i> <i class="texture_movie fa fa_big fa-film" title="Animated Texture" v-if="texture.frameCount > 1"></i> </div> <i class="material-icons texture_save_icon" v-bind:class="{clickable: !texture.saved}" v-on:click="texture.save()"> @@ -873,7 +915,7 @@ </i> <i class="material-icons texture_particle_icon" v-if="texture.particle">bubble_chart</i> <div class="texture_name">{{ texture.name }}</div> - <div class="texture_res">{{!Blockbench.entity_mode ? (texture.ratio == 1 ? texture.res + 'px' : (texture.res + 'px, ' + texture.frameCount+'f')) : (texture.res + ' x ' + texture.res*texture.ratio + 'px')}}</div> + <div class="texture_res">{{ texture.error ? texture.getErrorMessage() : (!Blockbench.entity_mode ? (texture.ratio == 1 ? texture.res + 'px' : (texture.res + 'px, ' + texture.frameCount+'f')) : (texture.res + ' x ' + texture.res*texture.ratio + 'px'))}}</div> </li> </ul> </div> @@ -882,9 +924,13 @@ <div id="options" class="panel selection_only"> <p class="tl">panel.options.angle</p> <div class="toolbar_wrapper rotation"></div> - <p class="tl">panel.options.origin</p> + <p class="tl">data.origin</p> <div class="toolbar_wrapper origin"></div> </div> + <div id="color" class="panel"> + <div id="main_colorpicker_preview"><div></div></div> + <input id="main_colorpicker"> + </div> <div id="outliner" class="panel grow"> <div class="toolbar_wrapper outliner"></div> <ul id="cubes_list" class="list"> @@ -951,6 +997,9 @@ <div class="f_right"> {{ Prop.fps }} FPS </div> + <div class="f_right" v-if="Prop.session"> + {{ Prop.connections }} Clients + </div> <div id="status_progress" v-if="Prop.progress" v-bind:style="{width: Prop.progress*100+'%'}"></div> </div> <script> diff --git a/index.php b/index.php index b5964a3d..74140e07 100644 --- a/index.php +++ b/index.php @@ -19,7 +19,7 @@ <script> if (typeof module === 'object') {window.module = module; module = undefined;}//jQuery Fix const isApp = typeof require !== 'undefined'; - const appVersion = '2.5.0'; + const appVersion = '2.6.0'; </script> <script src="lib/vue.min.js"></script> <script src="lib/vue_sortable.js"></script> @@ -30,6 +30,7 @@ <script src="lib/jimp.min.js"></script> <script src="lib/jszip.min.js"></script> <script src="lib/gif.js"></script> + <script src="lib/peer.min.js"></script> <script src="lib/spectrum.js"></script> <script src="lib/three.js"></script> <script src="lib/three_custom.js"></script> @@ -43,6 +44,7 @@ <script src="js/keyboard.js"></script> <script src="js/settings.js"></script> <script src="js/undo.js"></script> + <script src="js/edit_sessions.js"></script> <script type="text/javascript"> if (isApp === true) { @@ -86,6 +88,7 @@ ?></div> <div style="display: none;"></div> + <div id="blackout" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"></div> <div id="overlay_message_box" style="display: none;"> @@ -144,7 +147,7 @@ </div> <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> - <button type="button" class="" v-on:click="plugin.download()" v-else><i class="material-icons">add</i><span class="tl">dialog.plugins.install</span></button> + <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> <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> @@ -164,6 +167,35 @@ <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> </div> + <div class="dialog draggable paddinged" id="edit_sessions"> + <h2 class="dialog_handle tl">dialog.edit_session.title</h2> + + <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> + </div> + <div class="dialog draggable paddinged" id="toolbar_edit"> <h2 class="dialog_handle tl">dialog.toolbar_edit.title</h2> @@ -291,9 +323,7 @@ <div class="dialog draggable paddinged" id="scaling"> <h2 class="dialog_handle tl">dialog.scale.title</h2> - <div class="dialog_bar narrow"> - <label class="tl">dialog.scale.axis</label> - </div> + <label class="tl">dialog.scale.axis</label> <div class="dialog_bar" style="height: 32px;"> <input type="checkbox" class="toggle_panel" id="model_scale_x_axis" onchange="scaleAll()" checked> @@ -304,20 +334,28 @@ <label class="toggle_panel" for="model_scale_z_axis">Z</label> </div> - <div class="dialog_bar narrow"> - <label class="tl">dialog.scale.scale</label> + <label class="tl">data.origin</label> + <div class="dialog_bar"> + <label for="scaling_origin_x" class="inline_label tl">X</label> + <input type="number" id="scaling_origin_x" class="dark_bordered mediun_width" oninput="scaleAll()"> + <label for="scaling_origin_y" class="inline_label tl">Y</label> + <input type="number" id="scaling_origin_y" class="dark_bordered mediun_width" oninput="scaleAll()"> + <label for="scaling_origin_z" class="inline_label tl">Z</label> + <input type="number" id="scaling_origin_z" class="dark_bordered mediun_width" oninput="scaleAll()"> </div> + <label class="tl">dialog.scale.scale</label> <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> + <div class="dialog_bar narrow" id="scaling_clipping_warning"></div> <div class="dialog_bar"> - <button type="button" onclick="scaleAll(true)" class="large confirm_btn tl">dialog.scale.confirm</button> - <button type="button" class="large cancel_btn tl" onclick="cancelScaleAll()">dialog.cancel</button> - <button type="button" class="large hidden tl" id="scale_overflow_btn" onclick="scaleAllSelectOverflow()">dialog.scale.select_overflow</button> + <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> + <button type="button" class="hidden tl" id="scale_overflow_btn" onclick="scaleAllSelectOverflow()">dialog.scale.select_overflow</button> </div> <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> </div> @@ -437,7 +475,7 @@ <div class="dialog_bar"> - <button type="button" class="large tl confirm_btn cancel_btn" onclick="saveProjectSettings()">dialog.confirm</button> + <button type="button" class="large tl confirm_btn cancel_btn" onclick="saveProjectSettings();hideDialog();">dialog.confirm</button> <button type="button" class="large tl" id="entity_mode_convert" onclick="entityMode.convert()">dialog.project.to_entitymodel</button> </div> <div id="dialog_close_button" onclick="$('.dialog#'+open_dialog).find('.cancel_btn:not([disabled])').click()"><i class="material-icons">clear</i></div> @@ -481,9 +519,11 @@ <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'"> - <select v-model="setting.value" class="dark_bordered"> - <option v-for="(text, id) in setting.options" v-bind:value="id">{{ text }}</option> - </select> + <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> </template> </li> </ul> @@ -712,12 +752,15 @@ <div id="action_selector" v-if="open"> <input type="text" v-model="search_input"> <i class="material-icons" id="action_search_bar_icon">search</i> - <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> - {{ item.name }} - </li> - </ul> + <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> + {{ item.name }} + </li> + </ul> + <div class="small_text" v-if="actions[index]">{{ actions[index].description }}</div> + </div> </div> <header> @@ -879,16 +922,16 @@ > <div class="texture_icon_wrapper"> <img v-bind:texid="texture.id" v-bind:src="texture.source" class="texture_icon" width="48px" alt="[E]" v-if="texture.show_icon" /> - <i class="material-icons texture_error" title="Image Error" v-if="texture.error">error_outline</i> + <i class="material-icons texture_error" v-bind:title="texture.getErrorMessage()" v-if="texture.error">error_outline</i> <i class="texture_movie fa fa_big fa-film" title="Animated Texture" v-if="texture.frameCount > 1"></i> </div> - <i class="material-icons texture_particle_icon" v-if="texture.particle">bubble_chart</i> <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> + <i class="material-icons texture_particle_icon" v-if="texture.particle">bubble_chart</i> <div class="texture_name">{{ texture.name }}</div> - <div class="texture_res">{{!Blockbench.entity_mode ? (texture.ratio == 1 ? texture.res + 'px' : (texture.res + 'px, ' + texture.frameCount+'f')) : (texture.res + ' x ' + texture.res*texture.ratio + 'px')}}</div> + <div class="texture_res">{{ texture.error ? texture.getErrorMessage() : (!Blockbench.entity_mode ? (texture.ratio == 1 ? texture.res + 'px' : (texture.res + 'px, ' + texture.frameCount+'f')) : (texture.res + ' x ' + texture.res*texture.ratio + 'px'))}}</div> </li> </ul> </div> @@ -897,9 +940,13 @@ <div id="options" class="panel selection_only"> <p class="tl">panel.options.angle</p> <div class="toolbar_wrapper rotation"></div> - <p class="tl">panel.options.origin</p> + <p class="tl">data.origin</p> <div class="toolbar_wrapper origin"></div> </div> + <div id="color" class="panel"> + <div id="main_colorpicker_preview"><div></div></div> + <input id="main_colorpicker"> + </div> <div id="outliner" class="panel grow"> <div class="toolbar_wrapper outliner"></div> <ul id="cubes_list" class="list"> @@ -966,6 +1013,9 @@ <div class="f_right"> {{ Prop.fps }} FPS </div> + <div class="f_right" v-if="Prop.session"> + {{ Prop.connections }} Clients + </div> <div id="status_progress" v-if="Prop.progress" v-bind:style="{width: Prop.progress*100+'%'}"></div> </div> <script> diff --git a/js/TransformControls.js b/js/TransformControls.js index 779d031f..c8891eee 100644 --- a/js/TransformControls.js +++ b/js/TransformControls.js @@ -891,7 +891,7 @@ scope.keyframe = kf } } - Undo.initEdit({keyframes: scope.keyframe ? [scope.keyframe] : null}) + Undo.initEdit({keyframes: scope.keyframe ? [scope.keyframe] : []}) if (!scope.keyframe) { var ba = Animator.selected.getBoneAnimator() scope.keyframe = ba.addKeyframe(null, Timeline.second, channel); @@ -1083,9 +1083,6 @@ beforeFirstChange(event) var difference = value - (previousValue||0) - if (Toolbox.selected.transformerMode === 'scale') { - axis = 'x'; - } scope.keyframe.offset(axis, difference); scope.keyframe.select() @@ -1180,7 +1177,7 @@ updateSelection() } else if (Modes.id === 'animate') { - Undo.finishEdit('change keyframe') + Undo.finishEdit('change keyframe', {keyframes: [scope.keyframe]}) } else if (Modes.id === 'display') { Undo.finishEdit('edit display slot') diff --git a/js/actions.js b/js/actions.js index 45b38526..f3a9ec4e 100644 --- a/js/actions.js +++ b/js/actions.js @@ -19,6 +19,22 @@ class BarItem { this.condition = data.condition; this.nodes = [] this.toolbars = [] + //Key + this.category = data.category ? data.category : 'misc' + if (!data.private) { + if (data.keybind) { + this.default_keybind = data.keybind + } + if (Keybinds.stored[this.id]) { + this.keybind = new Keybind(Keybinds.stored[this.id]) + } else { + this.keybind = new Keybind(data.keybind) + } + this.keybind.setAction(this.id) + this.work_in_dialog = data.work_in_dialog === true + this.uses = 0; + Keybinds.actions.push(this) + } } conditionMet() { if (this.condition === undefined) { @@ -112,19 +128,6 @@ class Action extends BarItem { super(data) var scope = this; this.type = 'action' - this.category = data.category ? data.category : 'misc' - //Key - if (data.keybind) { - this.default_keybind = data.keybind - } - if (Keybinds.stored[this.id]) { - this.keybind = new Keybind(Keybinds.stored[this.id]) - } else { - this.keybind = new Keybind(data.keybind) - } - this.keybind.setAction(this.id) - this.work_in_dialog = data.work_in_dialog === true - this.uses = 0; //Icon this.icon = data.icon this.color = data.color @@ -148,7 +151,6 @@ class Action extends BarItem { if (data.linked_setting) { this.toggleLinkedSetting(false) } - Keybinds.actions.push(this) } trigger(event) { var scope = this; @@ -302,12 +304,16 @@ class NumSlider extends Widget { } } else { this.getInterval = function(event) { + event = event||false; return canvasGridSize(event.shiftKey, event.ctrlKey); }; } if (typeof data.getInterval === 'function') { this.getInterval = data.getInterval; } + if (this.keybind) { + this.keybind.shift = null; + } var scope = this; this.node = $( `<div class="tool wide widget nslide_tool"> <div class="tooltip">${this.name}</div> @@ -447,6 +453,17 @@ class NumSlider extends Widget { this.onAfter(difference) } } + trigger(event) { + if (typeof this.onBefore === 'function') { + this.onBefore() + } + var difference = this.getInterval(false) * event.shiftKey ? -1 : 1; + this.change(difference) + this.update() + if (typeof this.onAfter === 'function') { + this.onAfter(difference) + } + } setValue(value, trim) { if (typeof value === 'string') { value = parseFloat(value) @@ -503,9 +520,25 @@ class BarSlider extends Widget { if (typeof data.onChange === 'function') { this.onChange = data.onChange } + if (typeof data.onBefore === 'function') { + this.onBefore = data.onBefore + } + if (typeof data.onAfter === 'function') { + this.onAfter = data.onAfter + } $(this.node).children('input').on('input', function(event) { scope.change(event) }) + if (scope.onBefore) { + $(this.node).children('input').on('mousedown', function(event) { + scope.onBefore(event) + }) + } + if (scope.onAfter) { + $(this.node).children('input').on('change', function(event) { + scope.onAfter(event) + }) + } } change(event) { this.set( parseFloat( $(event.target).val() ) ) @@ -527,23 +560,24 @@ class BarSelect extends Widget { var scope = this; this.type = 'select' this.icon = 'list' - this.node = $('<div class="tool widget"><select class="dark_bordered"></select></div>').get(0) + this.node = $('<div class="tool widget bar_select"><select></select></div>').get(0) if (data.width) { $(this.node).children('select').css('width', data.width+'px') } + this.value = data.value + this.values = [] var select = $(this.node).find('select') if (data.options) { for (var key in data.options) { - if (data.options.hasOwnProperty(key)) { - if (!this.value) { - this.value = key - } - var name = data.options[key] - if (name === true) { - name = tl('action.'+this.id+'.'+key) - } - select.append('<option id="'+key+'">'+name+'</option>') + if (!this.value) { + this.value = key } + var name = data.options[key] + if (name === true) { + name = tl('action.'+this.id+'.'+key) + } + select.append(`<option id="${key}" ${key == this.value ? 'selected' : ''}>${name}</option>`) + this.values.push(key); } } this.addLabel(data.label) @@ -554,6 +588,26 @@ class BarSelect extends Widget { scope.change(event) }) } + trigger(event) { + var scope = this; + if (BARS.condition(scope.condition, scope)) { + if (event && event.type === 'click' && event.altKey && scope.keybind) { + var record = function() { + document.removeEventListener('keyup', record) + scope.keybind.record() + } + document.addEventListener('keyup', record, false) + return true; + } + var index = this.values.indexOf(this.value)+1 + if (index >= this.values.length) index = 0; + this.set(this.values[index]) + + scope.uses++; + return true; + } + return false; + } change(event) { this.set( $(event.target).find('option:selected').prop('id') ) if (this.onChange) { @@ -584,14 +638,20 @@ class BarText extends Widget { } } set(text) { + this.text = text; $(this.nodes).text(text) + return this; } update() { if (typeof this.onUpdate === 'function') { this.onUpdate() } + return this; + } + trigger(event) { + Blockbench.showQuickMessage(this.text) + return this; } - } class ColorPicker extends Widget { constructor(data) { @@ -646,7 +706,6 @@ class ColorPicker extends Widget { return this.value; } } - class Toolbar { constructor(data) { var scope = this; @@ -922,7 +981,8 @@ const BARS = { options: { move: true, scale: true - } + }, + category: 'edit' }) new Action({ id: 'swap_tools', @@ -1022,94 +1082,13 @@ const BARS = { keybind: new Keybind({key: 88, ctrl: true, shift: null}), click: function (event) {Clipbench.copy(event, true)} }) - new Action({ - id: 'duplicate', - icon: 'content_copy', - category: 'edit', - condition: () => (!display_mode && !Animator.open && (selected.length || selected_group)), - keybind: new Keybind({key: 68, ctrl: true}), - click: function () { - if (selected_group && (selected_group.matchesSelection() || selected.length === 0)) { - var cubes_before = elements.length - Undo.initEdit({outliner: true, cubes: [], selection: true}) - var g = selected_group.duplicate() - g.select().isOpen = true; - Undo.finishEdit('duplicate_group', {outliner: true, cubes: elements.slice().slice(cubes_before), selection: true}) - } else { - duplicateCubes(); - } - } - }) - new Action({ - id: 'delete', - icon: 'delete', - category: 'edit', - condition: () => (!display_mode && !Animator.open && selected.length), - keybind: new Keybind({key: 46}), - click: function () { - deleteCubes(); - } - }) - new Action({ - id: 'sort_outliner', - icon: 'sort_by_alpha', - category: 'edit', - click: function () { - Undo.initEdit({outliner: true}); - sortOutliner(); - Undo.finishEdit('sort_outliner') - } - }) - new Action({ - id: 'local_move', - icon: 'check_box', - category: 'edit', - linked_setting: 'local_move', - click: function () { - BarItems.local_move.toggleLinkedSetting() - updateSelection() - } - }) - new Action({ - id: 'select_window', - icon: 'filter_list', - category: 'edit', - condition: () => (!display_mode && !Animator.open), - keybind: new Keybind({key: 70, ctrl: true}), - click: function () { - showDialog('selection_creator') - $('#selgen_name').focus() - } - }) - new Action({ - id: 'invert_selection', - icon: 'swap_vert', - category: 'edit', - condition: () => (!display_mode && !Animator.open), - click: function () {invertSelection()} - }) - new Action({ - id: 'select_all', - icon: 'select_all', - category: 'edit', - condition: () => (!display_mode && !Animator.open), - keybind: new Keybind({key: 65, ctrl: true}), - click: function () {selectAll()} - }) - new Action({ - id: 'collapse_groups', - icon: 'format_indent_decrease', - category: 'edit', - condition: () => TreeElements.length > 0, - click: function () {collapseAllGroups()} - }) //Move Cube Keys new Action({ id: 'move_up', icon: 'arrow_upward', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 38, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(-1, 2, e)} }) @@ -1117,7 +1096,7 @@ const BARS = { id: 'move_down', icon: 'arrow_downward', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 40, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(1, 2, e)} }) @@ -1125,7 +1104,7 @@ const BARS = { id: 'move_left', icon: 'arrow_back', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 37, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(-1, 0, e)} }) @@ -1133,7 +1112,7 @@ const BARS = { id: 'move_right', icon: 'arrow_forward', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 39, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(1, 0, e)} }) @@ -1141,7 +1120,7 @@ const BARS = { id: 'move_forth', icon: 'keyboard_arrow_up', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 33, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(-1, 1, e)} }) @@ -1149,7 +1128,7 @@ const BARS = { id: 'move_back', icon: 'keyboard_arrow_down', category: 'transform', - condition: () => (selected.length && !open_menu), + condition: () => (selected.length && !open_menu && Modes.id === 'edit'), keybind: new Keybind({key: 34, ctrl: null, shift: null}), click: function (e) {moveCubesRelative(1, 1, e)} }) @@ -1218,21 +1197,18 @@ const BARS = { id: 'zoom_in', icon: 'zoom_in', category: 'view', - condition: isApp, click: function () {setZoomLevel('in')} }) new Action({ id: 'zoom_out', icon: 'zoom_out', category: 'view', - condition: isApp, click: function () {setZoomLevel('out')} }) new Action({ id: 'zoom_reset', icon: 'zoom_out_map', category: 'view', - condition: isApp, click: function () {setZoomLevel('reset')} }) @@ -1387,7 +1363,8 @@ const BARS = { Toolbars.keyframe = new Toolbar({ id: 'keyframe', children: [ - 'slider_keyframe_time' + 'slider_keyframe_time', + 'reset_keyframe' ], default_place: true }) @@ -1421,7 +1398,6 @@ const BARS = { children: [ 'brush_mode', 'fill_mode', - 'brush_color', 'slider_brush_size', 'slider_brush_opacity', 'slider_brush_softness' @@ -1978,11 +1954,12 @@ const MenuBar = { setup: function() { new BarMenu('file', [ 'project_window', - {name: 'menu.file.new', id: 'new', icon: 'insert_drive_file', children: [ + '_', + {name: 'menu.file.new', id: 'new', icon: 'insert_drive_file', condition: () => (!EditSession.active || EditSession.hosting), children: [ 'new_block_model', 'new_entity_model', ]}, - {name: 'menu.file.recent', id: 'recent', icon: 'history', condition: function() {return isApp && recent_projects.length}, children: function() { + {name: 'menu.file.recent', id: 'recent', icon: 'history', condition: function() {return isApp && recent_projects.length && (!EditSession.active || EditSession.hosting)}, children: function() { var arr = [] recent_projects.forEach(function(p) { switch (p.icon_id) { @@ -2012,11 +1989,13 @@ const MenuBar = { 'export_class_entity', 'export_optifine_part', 'export_optifine_full', - 'export_obj' + 'export_obj', + 'upload_sketchfab' ]}, 'save', '_', 'settings_window', + 'edit_session', 'update_window', 'donate', 'reload' @@ -2068,7 +2047,7 @@ const MenuBar = { ], () => (!display_mode && !Animator.open)) new BarMenu('filter', [ 'plugins_window', - '_' + '_', /* plaster optimize diff --git a/js/animations.js b/js/animations.js index 5dc54b5d..08bf4fca 100644 --- a/js/animations.js +++ b/js/animations.js @@ -27,6 +27,7 @@ class Animation { var ba = this.getBoneAnimator(group) var kfs = data.bones[key] if (kfs && ba) { + ba.keyframes.length = 0; kfs.forEach(kf_data => { var kf = new Keyframe(kf_data) ba.pushKeyframe(kf) @@ -98,8 +99,10 @@ class Animation { rename() { var scope = this; Blockbench.textPrompt('message.rename_animation', this.name, function(name) { - if (name) { + if (name && name !== scope.name) { + Undo.initEdit({animations: [scope]}) scope.name = name + Undo.finishEdit('rename animation') } }) return this; @@ -107,8 +110,10 @@ class Animation { editUpdateVariable() { var scope = this; Blockbench.textPrompt('message.animation_update_var', this.anim_time_update, function(name) { - if (name) { + if (name && name !== scope.anim_time_update) { + Undo.initEdit({animations: [this]}) scope.anim_time_update = name + Undo.finishEdit('change animation variable') } }) return this; @@ -140,18 +145,31 @@ class Animation { } Blockbench.dispatchEvent('display_animation_frame') } - add() { + add(undo) { + if (undo) { + Undo.initEdit({animations: []}) + } if (!Animator.animations.includes(this)) { Animator.animations.push(this) } + if (undo) { + this.select() + Undo.finishEdit('add animation', {animations: [this]}) + } return this; } - remove() { + remove(undo) { + if (undo) { + Undo.initEdit({animations: [this]}) + } if (Animator.selected === this) { Animator.selected = false } Animator.animations.remove(this) - Blockbench.dispatchEvent('remove_animation', {animation: this}) + if (undo) { + Undo.finishEdit('remove animation', {animation: null}) + } + Blockbench.dispatchEvent('remove_animation', {animations: [this]}) return this; } getMaxLength() { @@ -186,7 +204,7 @@ class Animation { animation.editUpdateVariable() }}, {name: 'generic.delete', icon: 'delete', click: function(animation) { - animation.remove() + animation.remove(true) }}, /* rename @@ -247,6 +265,7 @@ class BoneAnimator { } this.keyframes.push(keyframe) keyframe.parent = this; + TickUpdates.keyframes = true; return keyframe; } pushKeyframe(keyframe) { @@ -289,9 +308,11 @@ class BoneAnimator { displayScale(arr) { var bone = this.group.mesh if (arr) { - bone.scale.x = bone.scale.y = bone.scale.z = arr[0] ? arr[0] : 0.00001 + bone.scale.x = arr[0] ? arr[0] : 0.00001; + bone.scale.y = arr[1] ? arr[1] : 0.00001; + bone.scale.z = arr[2] ? arr[2] : 0.00001; } else { - bone.scale.x = bone.scale.y = bone.scale.z = 1 + bone.scale.x = bone.scale.y = bone.scale.z = 1; } return this; } @@ -329,12 +350,10 @@ class BoneAnimator { } else { let alpha = Math.lerp(before.time, after.time, time) result = [ - before.getLerp(after, 'x', alpha, allow_expression) + before.getLerp(after, 'x', alpha, allow_expression), + before.getLerp(after, 'y', alpha, allow_expression), + before.getLerp(after, 'z', alpha, allow_expression) ] - if (before.channel !== 'scale') { - result[1] = before.getLerp(after, 'y', alpha, allow_expression) - result[2] = before.getLerp(after, 'z', alpha, allow_expression) - } if (before.isQuaternion && after.isQuaternion) { result[3] = before.getLerp(after, 'q', alpha, allow_expression) } @@ -343,12 +362,10 @@ class BoneAnimator { let keyframe = result let method = allow_expression ? 'get' : 'calc' result = [ - keyframe[method]('x') + keyframe[method]('x'), + keyframe[method]('y'), + keyframe[method]('z') ] - if (keyframe.channel !== 'scale') { - result[1] = keyframe[method]('y') - result[2] = keyframe[method]('z') - } if (keyframe.isQuaternion) { result[3] = keyframe[method]('w') } @@ -405,7 +422,7 @@ class BoneAnimator { if (this.group && this.group.parent && this.group.parent !== 'root') { this.group.parent.openUp() } - Vue.nextTick(Timeline.update) + TickUpdates.keyframes = true; return this; } } @@ -424,8 +441,8 @@ class Keyframe { this.uuid = guid() if (typeof data === 'object') { this.extend(data) - if (this.channel === 'scale' && data.x === undefined) { - this.x = 1 + if (this.channel === 'scale' && data.x == undefined && data.y == undefined && data.z == undefined) { + this.x = this.y = this.z = 1; } } } @@ -507,9 +524,6 @@ class Keyframe { } } getArray() { - if (this.channel === 'scale') { - return this.get('x') - } var arr = [ this.get('x'), this.get('y'), @@ -659,6 +673,7 @@ class Keyframe { updateKeyframeSelection() } }, + 'copy', {name: 'generic.delete', icon: 'delete', click: function(keyframe) { keyframe.select({shiftKey: true}) removeSelectedKeyframes() @@ -696,8 +711,7 @@ function updateKeyframeSelection() { if (Timeline.selected.length && !multi_channel) { var first = Timeline.selected[0] $('#keyframe_type_label').text(tl('panel.keyframe.type', [tl('timeline.'+first.channel)] )) - $('#keyframe_bar_x').show() - $('#keyframe_bar_y, #keyframe_bar_z').toggle(first.channel !== 'scale') + $('#keyframe_bar_x, #keyframe_bar_y, #keyframe_bar_z').show() $('#keyframe_bar_w').toggle(first.channel === 'rotation' && first.isQuaternion) var values = [ @@ -747,6 +761,7 @@ function removeSelectedKeyframes() { } } updateKeyframeSelection() + Animator.preview() Undo.finishEdit('remove keyframes') } @@ -779,7 +794,7 @@ const Animator = { if (!Timeline.is_setup) { Timeline.setup() } - Timeline.update() + TickUpdates.keyframes = true; if (outlines.children.length) { outlines.children.length = 0 Canvas.updateAllPositions() @@ -859,7 +874,7 @@ const Animator = { }, buildFile: function(options) { if (typeof options !== 'object') { - options = {} + options = false } var animations = {} Animator.animations.forEach(function(a) { @@ -935,7 +950,7 @@ const Timeline = { setTimecode: function(time) { let m = Math.floor(time/60) let s = Math.floor(time%60) - let f = Math.floor((time%1) * 30) + let f = Math.floor((time%1) * 100) if ((s+'').length === 1) {s = '0'+s} if ((f+'').length === 1) {f = '0'+f} $('#timeline_corner').text(m + ':' + s + ':' + f) @@ -1036,7 +1051,7 @@ const Timeline = { var seconds = times[0]*60 + limitNumber(times[1], 0, 59) - + limitNumber(times[2]/30, 0, 29) + + limitNumber(times[2]/100, 0, 99) if (Math.abs(seconds-Timeline.second) > 1e-3 ) { Timeline.setTime(seconds, true) if (Animator.selected) { @@ -1045,25 +1060,47 @@ const Timeline = { } }) + $('#timeline_inner').on('mousewheel', function() { + if (event.ctrlKey) { + var offset = 1 - event.deltaY/600 + Timeline.vue._data.size = limitNumber(Timeline.vue._data.size * offset, 10, 1000) + this.scrollLeft *= offset + let l = (event.offsetX / this.clientWidth) * 500 * (event.deltaY<0?1:-0.2) + this.scrollLeft += l + } else { + this.scrollLeft += event.deltaY/2 + } + Timeline.updateSize() + event.preventDefault(); + }); + BarItems.slider_animation_speed.update() Timeline.is_setup = true Timeline.setTime(0) }, update: function() { //Draggable - $('#timeline_inner .keyframe').draggable({ + $('#timeline_inner .keyframe:not(.ui-draggable)').draggable({ axis: 'x', - distance: 10, + distance: 4, start: function(event, ui) { Undo.initEdit({keyframes: Timeline.keyframes, keep_saved: true}) var id = $(ui.helper).attr('id') + + var clicked = Timeline.vue._data.keyframes.findInArray('uuid', id) + if (clicked) { + clicked.select() + } + + + + var i = 0; - while (i < Timeline.vue._data.keyframes.length) { + for (var i = 0; i < Timeline.vue._data.keyframes.length; i++) { var kf = Timeline.vue._data.keyframes[i] - if (kf.uuid === id || kf.selected) { + if (kf.selected) { kf.time_before = kf.time } - i++; } }, drag: function(event, ui) { @@ -1102,6 +1139,7 @@ const Timeline = { Animator.preview() }, stop: function(event, ui) { + /* var id = $(ui.helper).attr('id') var i = 0; while (i < Timeline.vue._data.keyframes.length) { @@ -1110,7 +1148,7 @@ const Timeline = { kf.dragging = true } i++; - } + }*/ Undo.finishEdit('drag keyframes') } }) @@ -1200,7 +1238,6 @@ const Timeline = { var kf = bone.addKeyframe(false, Timeline.second, channel?channel:'rotation') kf.select() Undo.finishEdit('add_keyframe') - Vue.nextTick(Timeline.update) }, showMenu: function(event) { if (event.target.id === 'timeline_inner') { @@ -1220,12 +1257,12 @@ const Timeline = { Undo.initEdit({keyframes: bone.keyframes, keep_saved: true}) var kf = bone.addKeyframe(false, Math.round(time*30)/30, row === 2 ? 'scale' : (row === 1 ? 'position' : 'rotation')) kf.select().callMarker() - Vue.nextTick(Timeline.update) Undo.finishEdit('add_keyframe') } else { Blockbench.showQuickMessage('message.no_bone_selected') } - }} + }}, + 'paste' ]) } @@ -1257,7 +1294,7 @@ BARS.defineActions(function() { click: function () { var animation = new Animation({ name: 'animation.' + (Project.parent||'model') + '.new' - }).add().select() + }).add(true).rename() } }) @@ -1408,7 +1445,27 @@ BARS.defineActions(function() { Undo.initEdit({keyframes: Timeline.selected, keep_saved: true}) }, onAfter: function() { - Undo.finishEdit('edit keyframe') + Undo.finishEdit('move keyframes') + } + }) + new Action({ + id: 'reset_keyframe', + icon: 'replay', + category: 'animation', + condition: () => Animator.open, + click: function () { + Undo.initEdit({keyframes: Timeline.selected, keep_saved: true}) + Timeline.selected.forEach((kf) => { + var n = kf.channel === 'scale' ? 1 : 0; + kf.extend({ + x: n, + y: n, + z: n, + w: kf.isQuaternion ? 0 : undefined + }) + }) + Undo.finishEdit('reset keyframes') + Animator.preview() } }) diff --git a/js/api.js b/js/api.js index 421d182f..c65ff6b6 100644 --- a/js/api.js +++ b/js/api.js @@ -8,6 +8,7 @@ class API { this.selection = selected; this.flags = [] this.drag_handlers = {} + this.events = {} this.entity_mode = false if (isApp) { this.platform = process.platform @@ -25,6 +26,7 @@ class API { Undo.finishEdit() } reload() { + localStorage.removeItem('backup_model') if (isApp) { preventClosing = false Blockbench.flags.push('allow_reload') @@ -138,7 +140,7 @@ class API { var buttons = [] options.buttons.forEach(function(b, i) { - var btn = $('<button type="button" class="large">'+b+'</button>') + var btn = $('<button type="button">'+b+'</button>') btn.click(function(e) { hideDialog() setTimeout(function() { @@ -437,8 +439,18 @@ class API { } else { options.content = nativeImage.createFromPath(options.content).toPNG() } - } - if (options.custom_writer) { + } else if (options.savetype === 'zip') { + var fileReader = new FileReader(); + fileReader.onload = function(event) { + var buffer = Buffer.from(new Uint8Array(this.result)); + fs.writeFileSync(file_path, buffer) + if (cb) { + cb(file_path) + } + }; + fileReader.readAsArrayBuffer(options.content); + + } else if (options.custom_writer) { options.custom_writer(options.content, file_path) } else { fs.writeFileSync(file_path, options.content) @@ -469,35 +481,27 @@ class API { return this.flags[flag] } //Events - dispatchEvent(event_name, event) { - if (!this.listeners) { - return; - } - var i = 0; - while (i < this.listeners.length) { - if (this.listeners[i].name === event_name) { - this.listeners[i].callback(event) + dispatchEvent(event_name, data) { + var list = this.events[event_name] + if (!list) return; + for (var i = 0; i < list.length; i++) { + if (typeof list[i] === 'function') { + list[i](data) } - i++; } } addListener(event_name, cb) { - if (!this.listeners) { - this.listeners = [] + if (!this.events[event_name]) { + this.events[event_name] = [] } - this.listeners.push({name: event_name, callback: cb}) + this.events[event_name].safePush(cb) + } + on(event_name, cb) { + return Blockbench.addListener(event_name, cb) } removeListener(event_name, cb) { - if (!this.listeners) { - return; - } - var i = 0; - while (i < this.listeners.length) { - if (this.listeners[i].name === event_name && this.listeners[i].callback === cb) { - this.listeners.splice(i, 1) - } - i++; - } + if (!this.events[event_name]) return; + this.events[event_name].remove(cb); } //File Drag addDragHandler(id, options, cb) { @@ -525,11 +529,13 @@ function Dialog(settings) { var scope = this; this.title = settings.title this.lines = settings.lines + this.form = settings.form this.id = settings.id this.width = settings.width this.fadeTime = settings.fadeTime this.draggable = settings.draggable this.singleButton = settings.singleButton + this.buttons = settings.buttons if (!parseInt(settings.fadeTime)) this.fadeTime = 200 @@ -559,50 +565,149 @@ function Dialog(settings) { $(this.object).find('.cancel_btn:not([disabled])').click() } this.show = function() { - var jq_dialog = $('<div class="dialog paddinged" style="width: auto;" id="'+scope.id+'"><h2 class="dialog_handle">'+scope.title+'</h2></div>') + var jq_dialog = $(`<div class="dialog paddinged" style="width: auto;" id="${scope.id}"><h2 class="dialog_handle">${tl(scope.title)}</h2></div>`) scope.object = jq_dialog.get(0) var max_label_width = 0; - scope.lines.forEach(function(l) { - if (typeof l === 'object' && (l.label || l.widget)) { + if (scope.lines) { + scope.lines.forEach(function(l) { + if (typeof l === 'object' && (l.label || l.widget)) { - var bar = $('<div class="dialog_bar"></div>') - if (l.label) { - bar.append('<label class="name_space_left">'+tl(l.label)+(l.nocolon?'':':')+'</label>') - max_label_width = Math.max(getStringWidth(tl(l.label)), max_label_width) - } - if (l.node) { - bar.append(l.node) - } else if (l.widget) { - var widget = l.widget - if (typeof l.widget === 'string') { - widget = BarItems[l.widget] - } else if (typeof l.widget === 'function') { - widget = l.widget() + var bar = $('<div class="dialog_bar"></div>') + if (l.label) { + bar.append('<label class="name_space_left">'+tl(l.label)+(l.nocolon?'':':')+'</label>') + max_label_width = Math.max(getStringWidth(tl(l.label)), max_label_width) } - bar.append(widget.getNode()) - max_label_width = Math.max(getStringWidth(widget.name), max_label_width) + if (l.node) { + bar.append(l.node) + } else if (l.widget) { + var widget = l.widget + if (typeof l.widget === 'string') { + widget = BarItems[l.widget] + } else if (typeof l.widget === 'function') { + widget = l.widget() + } + bar.append(widget.getNode()) + max_label_width = Math.max(getStringWidth(widget.name), max_label_width) + } + jq_dialog.append(bar) + } else { + jq_dialog.append(l) } - jq_dialog.append(bar) - } else { - jq_dialog.append(l) - } - }) - if (max_label_width) { - document.styleSheets[0].insertRule('.dialog#'+this.id+' .dialog_bar label {width: '+(max_label_width+14)+'px}') + }) } - if (this.singleButton) { + if (scope.form) { + for (var form_id in scope.form) { + var data = scope.form[form_id] + if (data && Condition(data.condition)) { + var bar = $('<div class="dialog_bar"></div>') + if (data.label) { + bar.append(`<label class="name_space_left" for="${form_id}">${tl(data.label)+(data.nocolon?'':':')}</label>`) + max_label_width = Math.max(getStringWidth(tl(data.label)), max_label_width) + } + /* + type: +text + label + placeholder + */ + switch (data.type) { + default: + bar.append(`<input class="dark_bordered half" type="text" id="${form_id}" value="${data.value||''}" placeholder="${data.placeholder||''}">`) + break; + case 'textarea': + bar.append(`<textarea style="height: ${data.height||150}px;" id="${form_id}"></textarea>`) + break; + case 'text': + bar.append(`<p>${tl(data.text)}</p>`) + bar.addClass('small_text') + break; + case 'number': + bar.append(`<input class="dark_bordered half" type="number" id="${form_id}" value="${data.value||0}" min="${data.min}" max="${data.max}" step="${data.step||1}">`) + break; + case 'color': + if (!data.colorpicker) { + data.colorpicker = new ColorPicker({ + id: 'cp_'+form_id, + label: false, + private: true + }) + } + bar.append(data.colorpicker.getNode()) + break; + case 'checkbox': + bar.append(`<input type="checkbox" id="${form_id}"${data.value ? ' checked' : ''}>`) + break; + } + if (data.readonly) { + bar.find('input').attr('readonly', 'readonly') + } + jq_dialog.append(bar) + } + } + } + if (max_label_width) { + document.styleSheets[0].insertRule('.dialog#'+this.id+' .dialog_bar label {width: '+(max_label_width+8)+'px}') + } + if (this.buttons) { + + + var buttons = [] + + scope.buttons.forEach(function(b, i) { + var btn = $('<button type="button">'+b+'</button>') + buttons.push(btn) + }) + buttons[scope.confirmIndex||0].addClass('confirm_btn') + buttons[scope.cancelIndex||1].addClass('cancel_btn') + jq_dialog.append($('<div class="dialog_bar button_bar"></div>').append(buttons)) + + + + } else if (this.singleButton) { + jq_dialog.append('<div class="dialog_bar">' + '<button type="button" class="large cancel_btn confirm_btn"'+ (this.confirmEnabled ? '' : ' disabled') +'>'+tl('dialog.close')+'</button>' + '</div>') + } else { + jq_dialog.append(['<div class="dialog_bar">', '<button type="button" class="large confirm_btn"'+ (this.confirmEnabled ? '' : ' disabled') +'>'+tl('dialog.confirm')+'</button>', '<button type="button" class="large cancel_btn"'+ (this.cancelEnabled ? '' : ' disabled') +'>'+tl('dialog.cancel')+'</button>', '</div>'].join('')) + } jq_dialog.append('<div id="dialog_close_button" onclick="$(\'.dialog#\'+open_dialog).find(\'.cancel_btn:not([disabled])\').click()"><i class="material-icons">clear</i></div>') - $(this.object).find('.confirm_btn').click(this.onConfirm) - $(this.object).find('.cancel_btn').click(this.onCancel) + var confirmFn = function(e) { + + var result = {} + if (scope.form) { + for (var form_id in scope.form) { + var data = scope.form[form_id] + switch (data.type) { + default: + result[form_id] = jq_dialog.find('input#'+form_id).val() + break; + case 'text': break; + case 'textarea': + result[form_id] = jq_dialog.find('textarea#'+form_id).val() + break; + case 'number': + result[form_id] = parseFloat(jq_dialog.find('input#'+form_id).val())||0 + break; + case 'color': + result[form_id] = data.colorpicker.get(); + break; + case 'checkbox': + result[form_id] = jq_dialog.find('input#'+form_id).is(':checked') + break; + } + } + } + scope.onConfirm(result, e) + } + confirmFn.bind(this) + $(this.object).find('.confirm_btn').click(confirmFn) + $(this.object).find('.cancel_btn').click(() => {this.onCancel()}) //Draggable if (this.draggable !== false) { jq_dialog.addClass('draggable') diff --git a/js/app.js b/js/app.js index 6a81b732..22ca7b6b 100644 --- a/js/app.js +++ b/js/app.js @@ -14,11 +14,8 @@ var dialog_win = null, recent_projects= undefined; $(document).ready(function() { - if (electron.process.argv.length >= 2) { - if (electron.process.argv[1].substr(-5) == '.json') { - readFile(electron.process.argv[1], true) - } - } + + //Setup $('.open-in-browser').click((event) => { event.preventDefault(); shell.openExternal(event.target.href); @@ -33,6 +30,36 @@ $(document).ready(function() { if (__dirname.includes('C:\\xampp\\htdocs\\blockbench')) { Blockbench.addFlag('dev') } + + //Load Model + var model_loaded = false + if (electron.process.argv.length >= 2) { + var extension = pathToExtension(electron.process.argv[1]) + + if (['json', 'bbmodel', 'jem', 'jpm'].includes(extension)) { + Blockbench.read([electron.process.argv[1]], {}, (files) => { + + loadModel(files[0].content, files[0].path || files[0].path) + addRecentProject({name: pathToName(files[0].path, 'mobs_id'), path: files[0].path}) + model_loaded = true + }) + } + } + if (!model_loaded && localStorage.getItem('backup_model') && !currentwindow.webContents.second_instance) { + var backup_model = localStorage.getItem('backup_model') + localStorage.removeItem('backup_model') + Blockbench.showMessageBox({ + translateKey: 'recover_backup', + icon: 'fa-archive', + buttons: [tl('dialog.continue'), tl('dialog.cancel')], + confirm: 0, + cancel: 1 + }, function(result) { + if (result === 0) { + loadModel(backup_model, 'backup.bbmodel') + } + }) + } }); (function() { console.log('Electron '+process.versions.electron+', Node '+process.versions.node) @@ -44,7 +71,7 @@ function getLatestVersion(init) { $.getJSON('https://raw.githubusercontent.com/JannisX11/blockbench/master/package.json', (data) => { if (data.version) { latest_version = data.version - if (compareVersions(latest_version, appVersion) && init === true) { + if (compareVersions(latest_version, appVersion) && init === true && !open_dialog) { Blockbench.showMessageBox({ translateKey: 'update_notification', @@ -93,7 +120,7 @@ function addRecentProject(data) { icon_id = 2; } recent_projects.push({name: data.name, path: data.path, icon_id}) - if (recent_projects.length > 8) { + if (recent_projects.length > 12) { recent_projects.shift() } updateRecentProjects() @@ -187,7 +214,7 @@ function changeImageEditor(texture) { var dialog = new Dialog({ title: tl('message.image_editor.title'), id: 'image_editor', - lines: ['<div class="dialog_bar"><select class="dark_bordered input_wide">'+ + lines: ['<div class="dialog_bar"><select class="input_wide">'+ '<option id="ps">Photoshop</option>'+ '<option id="gimp">Gimp</option>'+ '<option id="pdn">Paint.NET</option>'+ @@ -329,18 +356,16 @@ function findEntityTexture(mob, return_path) { } else if (return_path === 'raw') { return ['entity', ...path.split('/')].join(osfs) } else { - if (fs.existsSync(texture_path + '.png')) { - var texture = new Texture({keep_size: true}).fromPath(texture_path + '.png').add() - } else if (fs.existsSync(texture_path + '.tga')) { - var texture = new Texture({keep_size: true}).fromPath(texture_path + '.tga').add() - - } else if (settings.default_path && settings.default_path.value) { - - texture_path = settings.default_path.value + osfs + 'entity' + osfs + path.split('/').join(osfs) - if (fs.existsSync(texture_path + '.png')) { - var texture = new Texture({keep_size: true}).fromPath(texture_path + '.png').add() - } else if (fs.existsSync(texture_path + '.tga')) { - var texture = new Texture({keep_size: true}).fromPath(texture_path + '.tga').add() + function tryItWith(extension) { + if (fs.existsSync(texture_path+'.'+extension)) { + var texture = new Texture({keep_size: true}).fromPath(texture_path+'.'+extension).add() + } + } + if (!tryItWith('png') && !tryItWith('tga')) { + if (settings.default_path && settings.default_path.value) { + + texture_path = settings.default_path.value + osfs + 'entity' + osfs + path.split('/').join(osfs) + tryItWith('png') || tryItWith('tga') } } } @@ -378,6 +403,11 @@ function saveFile(props) { BarItems.export_entity.trigger() } } + if (Blockbench.entity_mode && Prop.animation_path) { + Blockbench.writeFile(Prop.animation_path, { + content: autoStringify(Animator.buildFile()) + }) + } } function writeFileEntity(content, filepath) { @@ -388,15 +418,8 @@ function writeFileEntity(content, filepath) { try { data = fs.readFileSync(filepath, 'utf-8') } catch (err) {} - var obj = {} - if (content.bones && content.bones.length) { - var has_parents = false; - for (var i = 0; i < content.bones.length && !has_parents; i++) { - if (content.bones[i].parent) has_parents = true; - } - if (has_parents) { - obj.format_version = '1.8.0' - } + var obj = { + format_version: '1.10.0' } if (data) { try { @@ -547,6 +570,7 @@ function createBackup(init) { if (init || elements.length === 0) return; var model = buildBBModel() + localStorage.setItem('backup_model', model) var file_name = 'backup_'+d.getDate()+'.'+(d.getMonth()+1)+'.'+(d.getYear()-100)+'_'+d.getHours()+'.'+d.getMinutes() var file_path = folder_path+osfs+file_name+'.bbmodel' @@ -556,17 +580,6 @@ function createBackup(init) { } }) } -//Zoom -function setZoomLevel(mode) { - switch (mode) { - case 'in': Prop.zoom += 5; break; - case 'out': Prop.zoom -= 5; break; - case 'reset': Prop.zoom = 100; break; - } - var level = (Prop.zoom - 100) / 12 - currentwindow.webContents.setZoomLevel(level) - resizeWindow() -} //Close window.onbeforeunload = function() { if (preventClosing === true) { @@ -618,6 +631,7 @@ function showSaveDialog(close) { function closeBlockbenchWindow() { preventClosing = false; Blockbench.dispatchEvent('before_closing') + localStorage.removeItem('backup_model') if (!Blockbench.hasFlag('update_restart')) { return currentwindow.close(); diff --git a/js/blockbench.js b/js/blockbench.js index 6134e296..aba47e7a 100644 --- a/js/blockbench.js +++ b/js/blockbench.js @@ -24,6 +24,8 @@ const Prop = { fps: 0, zoom: 100, progress: 0, + session: false, + connections: 0, facing: 'north' } const Project = { @@ -140,7 +142,6 @@ function onVueSetup(func) { } onVueSetup.funcs.push(func) } - function canvasGridSize(shift, ctrl) { if (!shift && !ctrl) { return 16 / limitNumber(settings.edit_size.value, 1, 1024) @@ -156,7 +157,6 @@ function canvasGridSize(shift, ctrl) { return 16 / limitNumber(settings.shift_size.value, 1, 1024) } } - function updateNslideValues() { //if (!selected.length && (!Blockbench.entity_mode || !selected_group)) return; @@ -309,18 +309,6 @@ function unselectAll() { }) updateSelection() } -function invertSelection() { - elements.forEach(function(s) { - if (selected.includes(s)) { - selected.splice(selected.indexOf(s), 1) - } else { - selected.push(s) - } - }) - if (selected_group) selected_group.unselect() - updateSelection() - Blockbench.dispatchEvent('invert_selection') -} function createSelection() { if ($('#selgen_new').is(':checked')) { selected.length = 0 @@ -359,7 +347,6 @@ class Mode extends KeybindItem { this.condition = data.condition; this.onSelect = data.onSelect; this.onUnselect = data.onUnselect; - this.category = data.category; Modes.options[this.id] = this; } select() { @@ -407,17 +394,20 @@ BARS.defineActions(function() { new Mode({ id: 'edit', default_tool: 'move_tool', + category: 'navigate', keybind: new Keybind({key: 49}) }) new Mode({ id: 'paint', default_tool: 'brush_tool', + category: 'navigate', keybind: new Keybind({key: 50}) }) new Mode({ id: 'display', selectCubes: false, default_tool: 'move_tool', + category: 'navigate', keybind: new Keybind({key: 51}), condition: () => !Blockbench.entity_mode, onSelect: () => { @@ -430,6 +420,7 @@ BARS.defineActions(function() { new Mode({ id: 'animate', default_tool: 'move_tool', + category: 'navigate', keybind: new Keybind({key: 51}), condition: () => Blockbench.entity_mode, onSelect: () => { @@ -440,6 +431,17 @@ BARS.defineActions(function() { } }) }) +//Backup +setInterval(function() { + if (TreeElements.length || textures.length) { + try { + var model = buildBBModel() + localStorage.setItem('backup_model', model) + } catch (err) { + console.log('Unable to create backup. ', err) + } + } +}, 1e3*30) //Misc const TickUpdates = { Run: function() { @@ -451,6 +453,18 @@ const TickUpdates = { delete TickUpdates.selection; updateSelection() } + if (TickUpdates.main_uv) { + delete TickUpdates.main_uv; + main_uv.loadData() + } + if (TickUpdates.texture_list) { + delete TickUpdates.texture_list; + loadTextureDraggable(); + } + if (TickUpdates.keyframes) { + delete TickUpdates.keyframes; + Vue.nextTick(Timeline.update) + } } } const Screencam = { @@ -524,6 +538,7 @@ const Screencam = { if (!options.length) { options.length = 1000 } + var preview = quad_previews.current; var interval = options.fps ? (1000/options.fps) : 100 var gif = new GIF({ repeat: options.repeat, @@ -532,6 +547,11 @@ const Screencam = { }) var frame_count = (options.length/interval) + if (options.turnspeed) { + preview.controls.autoRotate = true; + preview.controls.autoRotateSpeed = options.turnspeed; + } + gif.on('finished', blob => { var reader = new FileReader() reader.onload = () => { @@ -550,7 +570,7 @@ const Screencam = { var frames = 0; var loop = setInterval(() => { var img = new Image() - img.src = quad_previews.current.canvas.toDataURL() + img.src = preview.canvas.toDataURL() img.onload = () => { gif.addFrame(img, {delay: interval}) } @@ -567,6 +587,9 @@ const Screencam = { if (Animator.open && Timeline.playing) { Timeline.pause() } + if (options.turnspeed) { + preview.controls.autoRotate = false; + } }, options.length) } } @@ -597,7 +620,7 @@ const Clipbench = { Clipbench.setCubes(selected) } if (cut) { - deleteCubes() + BarItems.delete.trigger() } } }, @@ -637,7 +660,7 @@ const Clipbench = { var img = clipboard.readImage() if (img) { var dataUrl = img.toDataURL() - var texture = new Texture({name: 'pasted', folder: 'blocks' }).fromDataURL(dataUrl).add().fillParticle() + var texture = new Texture({name: 'pasted', folder: 'blocks' }).fromDataURL(dataUrl).fillParticle().add(true) setTimeout(function() { texture.openMenu() },40) @@ -756,6 +779,13 @@ const Clipbench = { if (isApp) { clipboard.writeHTML(JSON.stringify({type: 'keyframes', content: Clipbench.keyframes})) } + }, + setText: function(text) { + if (isApp) { + clipboard.writeText(text) + } else { + document.execCommand('copy') + } } } diff --git a/js/display.js b/js/display.js index f46d7c5d..6dd0baf5 100644 --- a/js/display.js +++ b/js/display.js @@ -150,7 +150,7 @@ class refModel { case 'bow': this.onload = function() { var side = display_slot.includes('left') ? -1 : 1; - setDisplayArea(side*5.4, -5.6, 24.7, side*64, side*-25, side*55, 1,1,1) + setDisplayArea(side*4.2, -4.9, 25, -20, -19, -8, 1,1,1) } break; } @@ -1302,7 +1302,6 @@ enterDisplaySettings = function() { //Enterung Display Setting Mode, changes th buildGrid() setShading() DisplayMode.loadThirdRight() - Canvas.updateRenderSides() display_area.updateMatrixWorld() display_base.updateMatrixWorld() @@ -1482,6 +1481,7 @@ function loadDisp(key) { //Loads The Menu and slider values, common for all Radi DisplayMode.vue._data.slot = display[key] DisplayMode.slot = display[key] DisplayMode.updateDisplayBase() + Canvas.updateRenderSides() } DisplayMode.loadThirdRight = function() { //Loader diff --git a/js/edit_sessions.js b/js/edit_sessions.js new file mode 100644 index 00000000..b16da7e0 --- /dev/null +++ b/js/edit_sessions.js @@ -0,0 +1,239 @@ + +const EditSession = { + active: false, + hosting: false, + BBKey: '1h3sq3hoj6vfkh', + start: function() { + if (EditSession.active) return; + + EditSession.hosting = true; + Prop.session = true; + EditSession.setState(true); + var peer = EditSession.peer = new Peer({key: '1h3sq3hoj6vfkh'}); + EditSession.username = $('#edit_session_username').val() + + peer.on('open', (token) => { + $('#edit_session_token').val(token) + EditSession.token = token; + Clipbench.setText(token) + Blockbench.dispatchEvent('create_session', {peer, token}) + }) + peer.on('connection', (conn) => { + EditSession.initConnection(conn) + Prop.connections = Object.keys(peer.connections).length + console.log(tl('edit_session.joined', [conn.metadata.username])) + Blockbench.showQuickMessage(tl('edit_session.joined', [conn.metadata.username])) + //New Login + var model = buildBBModel({uuids: true, bitmaps: true, history: true}) + conn.on('open', function() { + Blockbench.dispatchEvent('user_joins_session', {conn}) + conn.send({ + type: 'init_model', + fromHost: EditSession.hosting, + sender: EditSession.peer.id, + data: model + }) + }) + conn.on('close', function() { + Blockbench.dispatchEvent('user_leaves_session', {conn}) + Blockbench.showQuickMessage(tl('edit_session.left', [conn.metadata.username])) + delete peer.connections[conn.peer] + Prop.connections = Object.keys(peer.connections).length + }) + }) + }, + join: function() { + if (EditSession.active) return; + + EditSession.hosting = false; + EditSession.peer = new Peer({key: '1h3sq3hoj6vfkh'}); + var token = $('#edit_session_token').val() + var username = $('#edit_session_username').val() + if (!token || !EditSession._matchToken(token)) { + Blockbench.showMessageBox({ + translateKey: 'invalid_session', + icon: 'cloud_off', + buttons: [tl('dialog.ok')], + }, result => { + showDialog('edit_sessions'); + }) + } + + EditSession.token = token; + var conn = EditSession.peer.connect(token, {metadata: {username: username}}); + + conn.on('error', (a, b) => { + Blockbench.showMessageBox({ + translateKey: 'invalid_session', + icon: 'cloud_off', + buttons: [tl('dialog.ok')], + }, result => { + showDialog('edit_sessions'); + }) + }) + conn.on('open', () => { + hideDialog() + EditSession.host = conn; + EditSession.setState(true); + EditSession.initConnection(conn) + Blockbench.dispatchEvent('join_session', {conn}) + }) + }, + quit: function() { + Blockbench.dispatchEvent('quit_session', {}) + if (EditSession.hosting) { + EditSession.sendAll('command', 'quit_session') + } else { + EditSession.host.close() + } + setTimeout(function() { + EditSession.setState(false) + EditSession.peer.destroy() + Prop.session = false; + Prop.connections = 0; + Blockbench.showQuickMessage('edit_session.quit_session', 1500) + }, 400) + + }, + setState: function(active) { + EditSession.active = active; + $('#edit_session_username, #edit_session_token').attr('readonly', active) + if (active) { + $('.edit_session_inactive').hide() + $('.edit_session_active').show() + $('#edit_session_status').text(EditSession.hosting ? tl('edit_session.hosting') : tl('edit_session.connected')) + $('#edit_session_copy_button .tooltip').text(tl('action.copy')) + } else { + EditSession.hosting = false; + $('.edit_session_active').hide() + $('.edit_session_inactive').show() + $('#edit_session_copy_button .tooltip').text(tl('action.paste')) + $('#edit_session_token').val('') + } + }, + dialog: function() { + showDialog('edit_sessions'); + if (!EditSession.active && isApp) { + var token = clipboard.readText() + if (EditSession._matchToken(token)) { + $('#edit_session_token').val(token) + } + var username = process.env.USERNAME + if (username) { + $('#edit_session_username').val(username) + } + } + }, + copyToken: function() { + var input = $('#edit_session_token') + if (EditSession.active) { + input.focus() + document.execCommand('selectAll') + document.execCommand('copy') + } else { + if (isApp) { + var token = clipboard.readText() + if (EditSession._matchToken(token)) { + $('#edit_session_token').val(token) + } + } else { + input.focus() + document.execCommand('selectAll') + document.execCommand('paste') + } + } + }, + initNewModel: function(force) { + if (EditSession.active && EditSession.hosting) { + var model = buildBBModel({uuids: true, bitmaps: true, raw: true}) + if (force) { + model.flag = 'force' + } + EditSession.sendAll('init_model', JSON.stringify(model)) + } + }, + + initConnection: function(conn) { + conn.on('data', EditSession.receiveData) + }, + sendAll: function(type, data) { + var tag = {type, data} + Blockbench.dispatchEvent('send_session_data', tag) + for (var key in EditSession.peer.connections) { + var conns = EditSession.peer.connections[key]; + conns.forEach(conn => { + conn.send({ + type: tag.type, + fromHost: EditSession.hosting, + sender: EditSession.peer.id, + data: tag.data + }); + }) + } + if (Blockbench.hasFlag('log_session')) { + console.log('Sent Data:', type, data) + } + }, + sendEdit: function(entry) { + var new_entry = { + before: omitKeys(entry.before, ['aspects']), + post: omitKeys(entry.post, ['aspects']), + save_history: entry.save_history, + action: entry.action + } + EditSession.sendAll('edit', JSON.stringify(new_entry)) + }, + receiveData: function(tag) { + if (Blockbench.hasFlag('log_session')) { + console.log('Received Data:', tag) + } + if (EditSession.hosting && !tag.hostOnly && Object.keys(EditSession.peer.connections).length > 1) { + //Redistribute + for (var id in EditSession.peer.connections) { + if (id !== tag.sender) { + EditSession.peer.connections[id][0].send(tag); + } + } + } + var data = tag.data; + if (typeof data === 'string' && (data.includes('"') || data.includes('['))) { + try { + data = tag.data = JSON.parse(data) + } catch (err) { + console.log(err) + return; + } + } + Blockbench.dispatchEvent('receive_session_data', tag) + + if (tag.type === 'edit') { + Undo.remoteEdit(data) + } else if (tag.type === 'init_model') { + force = data.flag === 'force'; + newProject(false, force) + loadBBModel(data) + } else if (tag.type === 'command') { + switch (data) { + case 'undo': Undo.undo(true); break; + case 'redo': Undo.redo(true); break; + case 'quit_session': EditSession.quit(); break; + } + } else if (tag.type === 'change_project_meta') { + for (var key in data) { + Project = data[key]; + } + } + }, + _matchToken: function(token) { + return !!(token.length === 16 && token.match(/[a-z0-9]{16}/)) + } +} + +BARS.defineActions(function() { + new Action({ + id: 'edit_session', + icon: 'people', + category: 'blockbench', + click: EditSession.dialog + }) +}) \ No newline at end of file diff --git a/js/element.js b/js/element.js index 6f6ae920..55a61f7a 100644 --- a/js/element.js +++ b/js/element.js @@ -13,8 +13,9 @@ var OutlinerButtons = { } Undo.initEdit({cubes: obj.forSelected(), outliner: true, selection: true}) obj.forSelected(function(cube) { - cube.remove(true) + cube.remove() }) + updateSelection() Undo.finishEdit('remove', {cubes: [], outliner: true, selection: true}) } }, @@ -150,8 +151,9 @@ class OutlinerElement { constructor(uuid) { this.uuid = uuid || guid() } - sortInBefore(element) { + sortInBefore(element, index_mod) { var index = -1; + index_mod = index_mod || 0; if (element.parent === 'root') { index = TreeElements.indexOf(element) @@ -169,7 +171,7 @@ class OutlinerElement { if (index < 0) arr.push(this) else { - arr.splice(index, 0, this) + arr.splice(index+index_mod, 0, this) } TickUpdates.outliner = true; @@ -217,21 +219,8 @@ class OutlinerElement { return this; } removeFromParent() { - var scope = this; - if (this.parent === 'root') { - TreeElements.forEach(function(s, i) { - if (s === scope) { - TreeElements.splice(i, 1) - } - }) - } else if (typeof this.parent === 'object') { - var childArray = this.parent.children - childArray.forEach(function(s, i) { - if (s === scope) { - childArray.splice(i, 1) - } - }) - } + this.getParentArray().remove(this); + return this; } getParentArray() { if (this.parent === 'root') { @@ -264,12 +253,14 @@ class OutlinerElement { $('#cubes_list').animate({ scrollTop: scroll_amount }, 200); + return this; } updateElement() { var scope = this; var old_name = this.name; scope.name = '_&/3%6-7A'; scope.name = old_name; + return this; } getDepth() { var d = 0; @@ -314,6 +305,7 @@ class OutlinerElement { scope.name = scope.old_name delete scope.old_name } + return this; } isIconEnabled(btn) { switch (btn.id) { @@ -598,9 +590,7 @@ class Cube extends OutlinerElement { } } delete Canvas.meshes[this.uuid] - if (selected.includes(this)) { - selected.splice(selected.indexOf(this), 1) - } + selected.remove(this) elements.splice(this.index, 1) if (Transformer.dragging) { outlines.remove(outlines.getObjectByName(this.uuid+'_ghost_outline')) @@ -1161,7 +1151,7 @@ class Group extends OutlinerElement { //Clear Old Group if (selected_group) selected_group.unselect() - if (event.shiftKey === true || event.ctrlKey === true) { + if (event.shiftKey !== true && event.ctrlKey !== true) { selected.length = 0 } //Select This Group @@ -1293,9 +1283,14 @@ class Group extends OutlinerElement { Undo.finishEdit('removed_group') } } - createUniqueName() { + createUniqueName(group_arr) { var scope = this; var others = getAllOutlinerGroups(); + if (group_arr && group_arr.length) { + group_arr.forEach(g => { + others.safePush(g) + }) + } var name = this.name.replace(/\d+$/, ''); function check(n) { for (var i = 0; i < others.length; i++) { @@ -1306,7 +1301,7 @@ class Group extends OutlinerElement { if (check(this.name)) { return this.name; } - for (var num = 2; num < 256; num++) { + for (var num = 2; num < 2e3; num++) { if (check(name+num)) { scope.name = name+num; return scope.name; @@ -1364,6 +1359,7 @@ class Group extends OutlinerElement { return this; } duplicate(destination) { + var copied_groups = []; function duplicateArray(g1, g2) { var array = g1.children var i = 0; @@ -1378,28 +1374,32 @@ class Group extends OutlinerElement { } } else { var copy = array[i].getChildlessCopy() - duplicateArray(array[i], copy) copy.addTo(g2) if (destination == 'cache') { copy.parent = undefined; } else if (Blockbench.entity_mode) { - copy.createUniqueName() + copy.createUniqueName(copied_groups) } + copied_groups.push(copy) + duplicateArray(array[i], copy) } i++; } } var base_group = this.getChildlessCopy() + if (destination !== 'cache') { + base_group.createUniqueName() + copied_groups.push(base_group) + } duplicateArray(this, base_group) base_group.parent = undefined; if (!destination) { - base_group.addTo(this.parent) + base_group.sortInBefore(this, 1).select() } else if (destination !== 'cache') { base_group.addTo(destination) } if (destination !== 'cache') { - base_group.createUniqueName() Canvas.updatePositions() TickUpdates.outliner = true; } @@ -1681,21 +1681,11 @@ function parseGroups(array, importGroup, startIndex) { } } //Outliner -function toggleOutlinerOptions(force) { - if (force === undefined) { - force = !$('.panel#outliner').hasClass('more_options') - } - if (force) { - $('.panel#outliner').addClass('more_options') - BarItems.outliner_toggle.setIcon('dns') - } else { - $('.panel#outliner').removeClass('more_options') - BarItems.outliner_toggle.setIcon('view_stream') - } -} function loadOutlinerDraggable() { function getOrder(loc, obj) { - if (obj.type === 'group') { + if (!obj) { + return; + } else if (obj.type === 'group') { if (loc < 8) return -1; if (loc > 24) return 1; } else { @@ -1788,15 +1778,17 @@ function loadOutlinerDraggable() { }) }) } -function collapseAllGroups() { - getAllOutlinerGroups().forEach(function(g) { - g.isOpen = false - var name = g.name - g.name = '_$X0v_' - g.name = name - }) -} function dropOutlinerObjects(item, target, event, order) { + if (item.type === 'group' && target && target.parent) { + var is_parent = false; + function iterate(g) { + if (!(is_parent = g === item) && g.parent.type === 'group') { + iterate(g.parent) + } + } + iterate(target) + if (is_parent) return; + } if (item.type === 'cube' && selected.includes( item )) { var items = selected.slice(); } else { @@ -1943,25 +1935,6 @@ function addGroup() { } //Misc -function deleteCubes(array) { - Undo.initEdit({cubes: selected, outliner: true, selection: true}) - if (selected_group) { - selected_group.remove(true) - return; - } - if (array == undefined) { - array = selected.slice(0) - } else if (array.constructor !== Array) { - array = [array] - } else { - array = array.slice(0) - } - array.forEach(function(s) { - s.remove(false) - }) - updateSelection() - Undo.finishEdit('delete') -} function duplicateCubes() { Undo.initEdit({cubes: [], outliner: true, selection: true}) selected.forEach(function(obj, i) { @@ -2007,14 +1980,6 @@ function stopRenameCubes(save) { Blockbench.removeFlag('renaming') } } -function sortOutliner() { - Undo.initEdit({outliner: true}) - if (TreeElements.length < 1) return; - TreeElements.sort(function(a,b) { - return sort_collator.compare(a.name, b.name) - }); - Undo.finishEdit('sort_outliner') -} function toggleCubeProperty(key) { var state = selected[0][key] if (typeof state === 'number') { @@ -2084,7 +2049,15 @@ BARS.defineActions(function() { category: 'edit', keybind: new Keybind({key: 115}), click: function () { - toggleOutlinerOptions() + + var state = !$('.panel#outliner').hasClass('more_options') + if (state) { + $('.panel#outliner').addClass('more_options') + BarItems.outliner_toggle.setIcon('dns') + } else { + $('.panel#outliner').removeClass('more_options') + BarItems.outliner_toggle.setIcon('view_stream') + } } }) new BarText({ @@ -2104,4 +2077,135 @@ BARS.defineActions(function() { } } }) + + new Action({ + id: 'duplicate', + icon: 'content_copy', + category: 'edit', + condition: () => (!display_mode && !Animator.open && (selected.length || selected_group)), + keybind: new Keybind({key: 68, ctrl: true}), + click: function () { + if (selected_group && (selected_group.matchesSelection() || selected.length === 0)) { + var cubes_before = elements.length + Undo.initEdit({outliner: true, cubes: [], selection: true}) + var g = selected_group.duplicate() + g.select().isOpen = true; + Undo.finishEdit('duplicate_group', {outliner: true, cubes: elements.slice().slice(cubes_before), selection: true}) + } else { + duplicateCubes(); + } + } + }) + new Action({ + id: 'delete', + icon: 'delete', + category: 'edit', + condition: () => (!display_mode && !Animator.open && (selected.length || selected_group)), + keybind: new Keybind({key: 46}), + click: function () { + + var array; + Undo.initEdit({cubes: selected, outliner: true, selection: true}) + if (selected_group) { + selected_group.remove(true) + return; + } + if (array == undefined) { + array = selected.slice(0) + } else if (array.constructor !== Array) { + array = [array] + } else { + array = array.slice(0) + } + array.forEach(function(s) { + s.remove(false) + }) + updateSelection() + Undo.finishEdit('delete') + } + }) + new Action({ + id: 'sort_outliner', + icon: 'sort_by_alpha', + category: 'edit', + click: function () { + Undo.initEdit({outliner: true}); + if (TreeElements.length < 1) return; + TreeElements.sort(function(a,b) { + return sort_collator.compare(a.name, b.name) + }); + Undo.finishEdit('sort_outliner') + } + }) + new Action({ + id: 'local_move', + icon: 'check_box', + category: 'edit', + linked_setting: 'local_move', + click: function () { + BarItems.local_move.toggleLinkedSetting() + updateSelection() + } + }) + new Action({ + id: 'element_colors', + icon: 'check_box', + category: 'edit', + linked_setting: 'outliner_colors', + click: function () { + BarItems.element_colors.toggleLinkedSetting() + updateSelection() + } + }) + new Action({ + id: 'select_window', + icon: 'filter_list', + category: 'edit', + condition: () => (!display_mode && !Animator.open), + keybind: new Keybind({key: 70, ctrl: true}), + click: function () { + showDialog('selection_creator') + $('#selgen_name').focus() + } + }) + new Action({ + id: 'invert_selection', + icon: 'swap_vert', + category: 'edit', + condition: () => (!display_mode && !Animator.open), + click: function () { + elements.forEach(function(s) { + if (selected.includes(s)) { + selected.splice(selected.indexOf(s), 1) + } else { + selected.push(s) + } + }) + if (selected_group) selected_group.unselect() + updateSelection() + Blockbench.dispatchEvent('invert_selection') + } + }) + new Action({ + id: 'select_all', + icon: 'select_all', + category: 'edit', + condition: () => (!display_mode && !Animator.open), + keybind: new Keybind({key: 65, ctrl: true}), + click: function () {selectAll()} + }) + new Action({ + id: 'collapse_groups', + icon: 'format_indent_decrease', + category: 'edit', + condition: () => TreeElements.length > 0, + click: function () { + getAllOutlinerGroups().forEach(function(g) { + g.isOpen = false + var name = g.name + g.name = '_$X0v_' + g.name = name + }) + } + }) }) diff --git a/js/interface.js b/js/interface.js index a6df59de..224e2457 100644 --- a/js/interface.js +++ b/js/interface.js @@ -60,6 +60,9 @@ class Panel { .click((event) => { setActivePanel(this.id) }) + .contextmenu((event) => { + setActivePanel(this.id) + }) .prepend(this.handle) } moveTo(ref_panel, before) { @@ -162,7 +165,7 @@ class ResizeLine { var Interface = { default_data: { - left_bar_width: 328, + left_bar_width: 338, right_bar_width: 300, quad_view_x: 50, quad_view_y: 50, @@ -254,6 +257,7 @@ function setupInterface() { } catch (err) {} $('.entity_mode_only').hide() + $('.edit_session_active').hide() $('.sidebar').droppable({ accept: 'h3', @@ -272,7 +276,7 @@ function setupInterface() { bottom: Toolbars.main_uv }, onResize: function() { - var size = limitNumber($(this.node).width()-4, 64, 1200) + var size = limitNumber($(this.node).width()-10, 64, 1200) size = Math.floor(size/16)*16 main_uv.setSize(size) } @@ -293,11 +297,38 @@ function setupInterface() { }) Interface.Panels.options = new Panel({ id: 'options', - condition: function() {return !display_mode && !Animator.open}, + condition: function() {return Modes.id === 'edit'}, toolbars: { - + rotation: Toolbars.rotation, + origin: Toolbars.origin, } }) + Interface.Panels.color = new Panel({ + id: 'color', + condition: () => Modes.id === 'paint', + toolbars: { + + }, + onResize: t => { + $('#main_colorpicker').spectrum('reflow'); + var h = $('.panel#color .sp-container.sp-flat').height()-20; + $('.panel#color .sp-palette').css('max-height', h+'px') + } + }) + Interface.Panels.color.picker = $('#main_colorpicker').spectrum({ + preferredFormat: "hex", + color: 'ffffff', + flat: true, + showAlpha: true, + showInput: true, + maxSelectionSize: 128, + showPalette: true, + palette: [], + localStorageKey: 'brush_color_palette', + move: function(c) { + $('#main_colorpicker_preview > div').css('background-color', c.toRgbString()) + } + }) Interface.Panels.outliner = new Panel({ id: 'outliner', condition: function() {return !display_mode}, @@ -310,9 +341,11 @@ function setupInterface() { menu: new Menu([ 'add_cube', 'add_group', + '_', 'sort_outliner', 'select_all', 'collapse_groups', + 'element_colors', 'outliner_toggle' ]) }) @@ -361,6 +394,7 @@ function setupInterface() { 'open_backup_folder', 'save' ]) + //$(document).contextmenu() //Tooltip Fix @@ -417,6 +451,9 @@ function setupInterface() { }) $(document).contextmenu(function(event) { if (!$(event.target).hasClass('allow_default_menu')) { + /*if (event.target.nodeName === 'INPUT' && $(event.target).is(':focus')) { + Interface.text_edit_menu.open(event, event.target) + }*/ return false; } }) @@ -438,20 +475,6 @@ function setupInterface() { eval(obj.attr('onmouseup')) }) - $('#timeline_inner').on('mousewheel', function() { - if (event.ctrlKey) { - var offset = 1 - event.deltaY/600 - Timeline.vue._data.size = limitNumber(Timeline.vue._data.size * offset, 10, 1000) - this.scrollLeft *= offset - let l = (event.offsetX / this.clientWidth) * 500 * (event.deltaY<0?1:-0.2) - this.scrollLeft += l - } else { - this.scrollLeft += event.deltaY/2 - } - Timeline.updateSize() - event.preventDefault(); - }); - //Mousemove $(document).mousemove(function(event) { mouse_pos.x = event.clientX @@ -548,11 +571,37 @@ function setProjectTitle(title) { $('title').text('Blockbench') } } +//Zoom +function setZoomLevel(mode) { + if (Prop.active_panel === 'uv') { + var zoom = main_uv.zoom + switch (mode) { + case 'in': zoom *= 1.5; break; + case 'out': zoom *= 0.66; break; + case 'reset': zoom = 1; break; + } + zoom = limitNumber(zoom, 1, 4) + main_uv.setZoom(zoom) + + } else if (isApp) { + switch (mode) { + case 'in': Prop.zoom += 5; break; + case 'out': Prop.zoom -= 5; break; + case 'reset': Prop.zoom = 100; break; + } + var level = (Prop.zoom - 100) / 12 + currentwindow.webContents.setZoomLevel(level) + resizeWindow() + } +} //Dialogs function showDialog(dialog) { var obj = $('.dialog#'+dialog) $('.dialog').hide(0) + if (open_menu) { + open_menu.hide() + } $('#blackout').fadeIn(200) obj.fadeIn(200) open_dialog = dialog diff --git a/js/io.js b/js/io.js index 1d3cba23..1b553a68 100644 --- a/js/io.js +++ b/js/io.js @@ -1,6 +1,6 @@ //New -function newProject(entity_mode) { - if (showSaveDialog()) { +function newProject(entity_mode, force) { + if (force || showSaveDialog()) { if (Toolbox.selected.id !== 'move_tool') BarItems.move_tool.select(); elements.length = 0; TreeElements.length = 1; @@ -95,7 +95,9 @@ function loadModel(data, filepath, add) { var extension = pathToExtension(filepath) if (extension === 'bbmodel') { - loadBBModel(model) + setTimeout(() => { + loadBBModel(model) + }, 8) } else if (extension === 'jpm') { loadJPMModel(model) } else if (extension === 'jem') { @@ -109,14 +111,13 @@ function loadModel(data, filepath, add) { } loadBlockModel(model, filepath, add) } - - loadTextureDraggable() - loadOutlinerDraggable() - Canvas.updateAll() Blockbench.removeFlag('importing') if (!add) { Prop.project_saved = true; } + if (!add) { + EditSession.initNewModel() + } } function loadBBModel(model) { if (!model.meta || !model.meta.format) { @@ -138,7 +139,6 @@ function loadBBModel(model) { } else { Blockbench.entity_mode = false; } - saveSettings() Project.name = model.name; if (model.geo_name) { Project.geometry_name = model.geo_name; @@ -155,8 +155,7 @@ function loadBBModel(model) { if (model.textures) { model.textures.forEach(tex => { - var tex_copy = new Texture(tex).add(false); - tex_copy.uuid = tex.uuid; + var tex_copy = new Texture(tex, tex.uuid).add(false); if (tex_copy.mode === 'link') { tex_copy.fromPath(tex.path) } else { @@ -166,7 +165,9 @@ function loadBBModel(model) { } if (model.cubes) { model.cubes.forEach(function(cube) { - base_cube = new Cube(cube).init(false) + base_cube = new Cube() + if (cube.uuid) base_cube.uuid = cube.uuid; + base_cube.extend(cube).init(false) for (var face in base_cube.faces) { if (!model.meta.box_uv) { var texture = textures[cube.faces[face].texture] @@ -182,15 +183,26 @@ function loadBBModel(model) { } if (model.outliner) { parseGroups(model.outliner) + if (model.meta.bone_rig) { + Canvas.updateAllBones() + Canvas.updateAllPositions() + } } if (model.animations) { model.animations.forEach(ani => { - var base_ani = new Animation(ani).add(); + var base_ani = new Animation() + base_ani.uuid = ani.uuid; + base_ani.extend(ani).add(); }) } if (model.display !== undefined) { DisplayMode.loadJSON(model.display) } + if (model.history) { + Undo.history = model.history.slice() + Undo.index = model.history_index; + } + updateSelection() } function loadBlockModel(model, filepath, add) { if (!model.elements && !model.parent && !model.display && !model.textures) { @@ -209,7 +221,10 @@ function loadBlockModel(model, filepath, add) { var previous_length = add ? elements.length : 0 var previous_texture_length = add ? textures.length : 0 + var new_cubes = []; + var new_textures = []; if (add) { + Undo.initEdit({cubes: new_cubes, outliner: true, textures: new_textures}) Prop.added_models++; var import_group = new Group(pathToName(filepath, false)) } @@ -219,6 +234,7 @@ function loadBlockModel(model, filepath, add) { DisplayMode.loadJSON(model.display) } var texture_ids = {} + var texture_paths = {} if (model.textures) { //Create Path Array to fetch textures var path_arr = filepath.split(osfs) @@ -226,22 +242,23 @@ function loadBlockModel(model, filepath, add) { path_arr.splice(-index) var texture_arr = model.textures - var paths = {} for (var tex in texture_arr) { if (texture_arr.hasOwnProperty(tex)) { if (tex != 'particle') { var t = new Texture({id: tex}).fromJavaLink(texture_arr[tex], path_arr.slice()).add(false) - paths[texture_arr[tex]] = texture_ids[tex] = t + texture_paths[texture_arr[tex]] = texture_ids[tex] = t + new_textures.push(t); } } } if (texture_arr.particle) { - if (paths[texture_arr.particle]) { - paths[texture_arr.particle].enableParticle() + if (texture_paths[texture_arr.particle]) { + texture_paths[texture_arr.particle].enableParticle() } else { var t = new Texture({id: 'particle'}).fromJavaLink(texture_arr[tex], path_arr.slice()).add(false).enableParticle() - texture_ids.particle = t; + texture_paths[texture_arr[tex]] = texture_ids.particle = t; + new_textures.push(t); } } //Get Rid Of ID overlapping @@ -278,10 +295,22 @@ function loadBlockModel(model, filepath, add) { if (obj.faces[face].texture === '#missing') { } else if (obj.faces[face].texture) { - var t = texture_ids[obj.faces[face].texture.replace(/^#/, '')] - if (t instanceof Texture) { - base_cube.faces[face].texture = t.uuid; + var id = obj.faces[face].texture.replace(/^#/, '') + var t = texture_ids[id] + + if (t instanceof Texture === false) { + if (texture_paths[obj.faces[face].texture]) { + var t = texture_paths[obj.faces[face].texture] + if (t.id === 'particle') { + t.extend({id: id, name: '#'+id}).loadEmpty(3) + } + } else { + var t = new Texture({id: id, name: '#'+id}).add(false).loadEmpty(3) + texture_ids[id] = t + new_textures.push(t); + } } + base_cube.faces[face].texture = t.uuid; } if (obj.faces[face].tintindex !== undefined) { base_cube.faces[face].tint = true; @@ -294,7 +323,6 @@ function loadBlockModel(model, filepath, add) { } else { base_cube.autouv = 0; } - elements.push(base_cube); if (!add) { TreeElements.push(base_cube) base_cube.parent = 'root' @@ -302,6 +330,8 @@ function loadBlockModel(model, filepath, add) { import_group.children.push(base_cube) base_cube.parent = import_group } + base_cube.init() + new_cubes.push(base_cube); }) } if (model.groups && model.groups.length > 0) { @@ -326,18 +356,17 @@ function loadBlockModel(model, filepath, add) { from: [0, 0, 7.5], to: [16, 16, 7.8], faces: { - north: {uv: [16,0,0,16], texture: 'layer0'}, - south: {uv: [16,0,16,0], texture: 'layer0'}, + north: {uv: [16,0,0,16], texture: textures[0].uuid || null}, + south: {uv: [0,0,16,16], texture: textures[0].uuid || null}, east: {uv: [0,0,0,0], texture: null}, west: {uv: [0,0,0,0], texture: null}, - up: {uv: [0,0,0,0], texture: null}, + up: {uv: [0,0,0,0], texture: null}, down: {uv: [0,0,0,0], texture: null}, }, autouv: 0, export: false - }) - elements.push(base_cube); - base_cube.addTo() + }).init() + new_cubes.push(base_cube); } else if (!model.elements && model.parent) { Blockbench.showMessageBox({ translateKey: 'child_model_only', @@ -345,6 +374,7 @@ function loadBlockModel(model, filepath, add) { message: tl('message.child_model_only.message', [model.parent]) }) } + updateSelection() //Set Parent if (model.parent !== undefined) { @@ -354,8 +384,15 @@ function loadBlockModel(model, filepath, add) { if (model.ambientocclusion === false) { Project.ambientocclusion = false; } + if (add) { + Undo.finishEdit('add block model') + } } -function loadJPMModel(model) { +function loadJPMModel(model, add) { + var new_cubes = []; + if (add) { + Undo.initEdit({cubes: new_cubes, outliner: true}) + } function addSubmodel(submodel) { if (submodel.boxes) { submodel.boxes.forEach(function(box) { @@ -382,9 +419,8 @@ function loadJPMModel(model) { down: {uv: box.uvDown}, }, rotation: submodel.rotate - }) - elements.push(base_cube); - TreeElements.push(base_cube) + }).init() + new_cubes.push(base_cube); } }) } @@ -392,10 +428,13 @@ function loadJPMModel(model) { submodel.submodels.forEach(addSubmodel) } } + if (add) { + Undo.finishEdit('add jpm model') + } addSubmodel(model) Canvas.updateAll() } -function loadJEMModel(model) { +function loadJEMModel(model, add) { entityMode.join() if (model.textureSize) { Project.texture_width = parseInt(model.textureSize[0]) @@ -498,7 +537,6 @@ function loadEntityModelFile(data) { if (pe_list && pe_list._data) { pe_list._data.search_text = '' } - saveSettings() function rotateOriginCoord(pivot, y, z) { return [ @@ -751,6 +789,7 @@ function loadEntityModel(data) { if (isApp && Project.parent) { findEntityTexture(Project.parent) } + EditSession.initNewModel() } var Extruder = { drawImage: function(path) { @@ -978,22 +1017,26 @@ function buildBBModel(options) { if (!cube.rotation.allEqual(0)) el.rotation = cube.rotation; if (!cube.origin.allEqual(0)) el.origin = cube.origin; if (!cube.uv_offset.allEqual(0)) el.uv_offset = cube.uv_offset; - if (!model.meta.box_uv) { el.faces = {} for (var face in cube.faces) { el.faces[face] = cube.faces[face].getSaveCopy() } } + el.uuid = cube.uuid model.cubes.push(el) }) - model.outliner = compileGroups() + model.outliner = compileGroups(true) model.textures = []; textures.forEach(tex => { var t = tex.getUndoCopy(); delete t.selected; + if (options.bitmaps) { + t.source = 'data:image/png;base64,'+tex.getBase64() + t.mode = 'bitmap' + } model.textures.push(t); }) @@ -1004,7 +1047,6 @@ function buildBBModel(options) { }) } - if (!Blockbench.entity_mode && Object.keys(display).length >= 1) { var new_display = {} var entries = 0; @@ -1019,10 +1061,24 @@ function buildBBModel(options) { model.display = new_display } } + + if (options.history) { + model.history = []; + Undo.history.forEach(h => { + var e = { + before: omitKeys(h.before, ['aspects']), + post: omitKeys(h.post, ['aspects']), + action: h.action + } + model.history.push(e); + }) + model.history_index = Undo.index; + } + if (options.raw) { - return model + return model; } else { - return JSON.stringify(model) + return JSON.stringify(model); } } function buildBlockModel(options) { @@ -1164,9 +1220,12 @@ function buildBlockModel(options) { textures.forEach(function(t, i){ if (!textures_used.includes(t) && !isTexturesOnlyModel) return; - texturesObj[t.id] = t.javaTextureLink(options.backup) + var link = t.javaTextureLink() + if (t.id !== link.replace(/^#/, '')) { + texturesObj[t.id] = link + } if (t.particle) { - texturesObj.particle = t.javaTextureLink(options.backup) + texturesObj.particle = link } if (t.mode === 'bitmap') { hasUnsavedTextures = true @@ -1292,33 +1351,34 @@ function buildEntityModel(options) { bone.material = g.material } //Cubes - if (g.children && g.children.length) { - bone.cubes = [] - var i = 0; - while (i < g.children.length) { - var s = g.children[i] - if (s !== undefined && s.type === 'cube' && s.export !== false) { - var cube = new oneLiner() - cube.origin = s.from.slice() - cube.size = s.size() - cube.origin[0] = -(cube.origin[0] + cube.size[0]) - cube.uv = s.uv_offset - if (s.inflate && typeof s.inflate === 'number') { - cube.inflate = s.inflate - } - if (s.mirror_uv === !bone.mirror) { - cube.mirror = s.mirror_uv - } - //Visible Bounds - var mesh = s.mesh - if (mesh) { - visible_box.expandByObject(mesh) - } - bone.cubes.push(cube) - cube_count++; + var cubes = [] + var i = 0; + while (i < g.children.length) { + var s = g.children[i] + if (s !== undefined && s.type === 'cube' && s.export !== false) { + var cube = new oneLiner() + cube.origin = s.from.slice() + cube.size = s.size() + cube.origin[0] = -(cube.origin[0] + cube.size[0]) + cube.uv = s.uv_offset + if (s.inflate && typeof s.inflate === 'number') { + cube.inflate = s.inflate } - i++; + if (s.mirror_uv === !bone.mirror) { + cube.mirror = s.mirror_uv + } + //Visible Bounds + var mesh = s.mesh + if (mesh) { + visible_box.expandByObject(mesh) + } + cubes.push(cube) + cube_count++; } + i++; + } + if (cubes.length) { + bone.cubes = cubes } bones.push(bone) }) @@ -1660,6 +1720,75 @@ function buildOBJModel(name) { scene.position.set(-8,-8,-8) return content; } +function uploadSketchfabModel() { + + var dialog = new Dialog({ + id: 'sketchfab_uploader', + title: 'Upload Sketchfab Model', + width: 540, + form: { + token: {label: 'dialog.sketchfab_uploader.token', value: settings.sketchfab_token.value}, + about_token: {type: 'text', text: 'dialog.sketchfab_uploader.about_token'}, + name: {label: 'dialog.sketchfab_uploader.name'}, + description: {label: 'dialog.sketchfab_uploader.description', type: 'textarea'}, + tags: {label: 'dialog.sketchfab_uploader.tags', placeholder: 'Tag1 Tag2'}, + }, + onConfirm: function(formResult) { + if (formResult.token && !formResult.name) { + Blockbench.showQuickMessage('message.sketchfab.name_or_token', 1800) + return; + } + if (!formResult.tags.split(' ').includes('blockbench')) { + formResult.tags += ' blockbench'; + } + var data = new FormData() + data.append('token', formResult.token) + data.append('name', formResult.name) + data.append('description', formResult.description) + data.append('tags', formResult.tags) + + settings.sketchfab_token.value = formResult.token + + var archive = new JSZip(); + var model_data = buildOBJModel('model') + archive.file('model.obj', model_data.obj) + archive.file('model.mtl', model_data.mtl) + for (var key in model_data.images) { + var tex = model_data.images[key]; + if (tex) { + archive.file(pathToName(tex.name) + '.png', tex.getBase64(), {base64: true}); + } + } + + archive.generateAsync({type: 'blob'}).then(blob => { + + var file = new File([blob], 'model.zip', {type: 'application/x-zip-compressed'}) + data.append('modelFile', file) + + $.ajax({ + url: 'https://api.sketchfab.com/v3/models', + data: data, + cache: false, + contentType: false, + processData: false, + type: 'POST', + success: function(response) { + Blockbench.showQuickMessage('message.sketchfab.success', 1500) + Blockbench.openLink('https://sketchfab.com/models/'+response.uid) + }, + error: function(response) { + Blockbench.showQuickMessage('message.sketchfab.error', 1500) + console.error(response); + } + }) + }) + + dialog.hide() + } + }) + dialog.show() +} + function compileJSON(object, options) { var output = '' if (typeof options !== 'object') options = {} @@ -1770,6 +1899,22 @@ function autoParseJSON(data, feedback) { return data; } +function saveProjectSettings() { + if (Blockbench.entity_mode) { + main_uv.setGrid() + if (uv_dialog.editors) { + uv_dialog.editors.single.setGrid() + } + if (entityMode.old_res.x !== Project.texture_width || entityMode.old_res.y !== Project.texture_height) { + entityMode.setResolution() + Undo.finishEdit('changed resolution') + } + if (EditSession.active && EditSession.hosting) { + EditSession.sendAll('change_project_meta', JSON.stringify(Project)); + } + } +} + BARS.defineActions(function() { //New new Action({ @@ -1777,7 +1922,11 @@ BARS.defineActions(function() { icon: 'insert_drive_file', category: 'file', keybind: new Keybind({key: 78, ctrl: true}), - click: function () {newProject()} + click: function () { + if (newProject()) { + EditSession.initNewModel() + } + } }) new Action({ id: 'new_entity_model', @@ -1787,6 +1936,7 @@ BARS.defineActions(function() { click: function () { if (newProject(true)) { showDialog('project_settings'); + EditSession.initNewModel() } } }) @@ -1796,6 +1946,7 @@ BARS.defineActions(function() { icon: 'assessment', category: 'file', keybind: new Keybind({key: 79, ctrl: true}), + condition: () => (!EditSession.active || EditSession.hosting), click: function () { Blockbench.import({ extensions: ['json', 'jem', 'jpm', 'bbmodel'], @@ -1976,7 +2127,6 @@ BARS.defineActions(function() { var content = buildEntityModel() } archive.file((Project.name||'model')+'.json', content) - var texfolder = archive.folder('textures'); textures.forEach(tex => { if (tex.mode === 'bitmap') { texfolder.file(pathToName(tex.name) + '.png', tex.source.replace('data:image/png;base64,', ''), {base64: true}); @@ -1995,4 +2145,12 @@ BARS.defineActions(function() { }) } }) + new Action({ + id: 'upload_sketchfab', + icon: 'fa-cube', + category: 'file', + click: function(ev) { + uploadSketchfabModel() + } + }) }) diff --git a/js/keyboard.js b/js/keyboard.js index 44889709..b2673f13 100644 --- a/js/keyboard.js +++ b/js/keyboard.js @@ -253,7 +253,7 @@ $(document).keydown(function(e) { if (e.ctrlKey === true && e.which == 73 && isApp) { electron.getCurrentWindow().toggleDevTools() used = true - } else if (e.which === 18 && Toolbox.selected.alt_tool && !Toolbox.original) { + } else if (e.which === 18 && Toolbox.selected.alt_tool && !Toolbox.original && !open_interface) { //Alt Tool var orig = Toolbox.selected; var alt = BarItems[Toolbox.selected.alt_tool] diff --git a/js/language.js b/js/language.js index cd1d39b8..7ffa52cc 100644 --- a/js/language.js +++ b/js/language.js @@ -32,10 +32,12 @@ const Language = { ja: '\u65E5\u672C\u8A9E (Japanese)',//日本語 nl: 'Nederlands (Dutch)', pl: 'Polski (Polish)', + pt: 'Portugu\u00EAs (Portuguese)', ru: '\u0440\u0443\u0441\u0441\u043A\u0438\u0439 (Russian)', sv: 'Svenska (Swedish)', zh: '\u4e2d\u6587 (Chinese)',//中文 - } + }, + toString: () => Language.code } function getStringWidth(string, size) { var a = $('<label style="position: absolute">'+string+'</label>') @@ -61,7 +63,7 @@ function loadLanguage() { } $.ajax({ dataType: "json", - url: 'lang/'+Language.code+'.json', + url: 'lang/'+Language+'.json', //data: data, //async: false, success: function(data) { diff --git a/js/molang.js b/js/molang.js index 00fd46fa..f0eabdad 100644 --- a/js/molang.js +++ b/js/molang.js @@ -250,7 +250,7 @@ function previewVariableValue(name, time) { return 1 } else if (name === 'false') { return 0 - } else if (name === 'global.anim_time' || name === 'time' || name === 'query.life_time' ) { + } else if (name === 'global.anim_time' || name === 'query.anim_time' || name === 'time' || name === 'query.life_time' ) { return time } else { var inputs = $('#var_placeholder_area').val().split('\n') diff --git a/js/painter.js b/js/painter.js index 779fc2d3..95304f0d 100644 --- a/js/painter.js +++ b/js/painter.js @@ -162,7 +162,7 @@ class BBPainter { b: px[2], a: px[3]/256 }) - BarItems.brush_color.set(t) + ColorPanel.set(t) }) } useBrush(texture, x, y, uvTag, no_update) { @@ -174,7 +174,7 @@ class BBPainter { var ctx = canvas.getContext('2d') ctx.save() - var color = BarItems.brush_color.get().toRgb();//.toRgbString() + var color = ColorPanel.get().toRgb();//.toRgbString() var size = BarItems.slider_brush_size.get(); var softness = BarItems.slider_brush_softness.get()/100; var b_opacity = BarItems.slider_brush_opacity.get()/100; @@ -196,15 +196,15 @@ class BBPainter { if (rect[t] > rect[t+2]) { [rect[t], rect[t+2]] = [rect[t+2], rect[t]] } - rect[t] = Math.floor(rect[t]) - rect[t+2] = Math.ceil(rect[t+2]) + rect[t] = Math.round(rect[t]) + rect[t+2] = Math.round(rect[t+2]) } var [w, h] = [rect[2] - rect[0], rect[3] - rect[1]] ctx.rect(rect[0], rect[1], w, h) if (tool === 'fill_tool') { - ctx.fillStyle = BarItems.brush_color.get().toRgbString() + ctx.fillStyle = ColorPanel.get().toRgbString() var fill_mode = BarItems.fill_mode.get() var cube = Painter.current.cube; @@ -462,61 +462,43 @@ class BBPainter { }) } addBitmapDialog() { - var lines = [] - - lines.push({label: 'dialog.create_texture.name', node: '<input class="dark_bordered half" type="text" id="bitmap_name">'}) - lines.push({label: 'dialog.create_texture.folder', node: '<input class="dark_bordered half" type="text" id="bitmap_folder">'}) - if (elements.length > 0) { - lines.push({label: 'dialog.create_texture.template', node: '<input type="checkbox" id="bitmap_doTemplate">'}) - lines.push({label: 'dialog.create_texture.compress', node: '<input type="checkbox" id="bitmap_compressTemplate">'}) - } - lines.push({widget: Painter.background_color}) - lines.push({label: 'dialog.create_texture.resolution', node: '<input class="dark_bordered" style="width:72px" type="number" id="bitmap_resolution">'}) - - var dialog = new Dialog({ id: 'add_bitmap', title: tl('dialog.create_texture.title'), - draggable: true, - lines: lines, - onConfirm: function() { - Painter.addBitmapFromDialog() + form: { + bitmap_name: {label: 'dialog.create_texture.name'}, + bitmap_folder: {label: 'dialog.create_texture.folder'}, + bitmap_doTemplate: {label: 'dialog.create_texture.template', type: 'checkbox', condition: elements.length}, + bitmap_compressTemplate: {label: 'dialog.create_texture.compress', type: 'checkbox'}, + bitmap_power: {label: 'dialog.create_texture.power', type: 'checkbox', value: true}, + background_color: {type: 'color', colorpicker: Painter.background_color}, + bitmap_resolution: {label: 'dialog.create_texture.resolution', type: 'number', value: 16}, + }, + onConfirm: function(results) { + + Painter.addBitmap({ + res: limitNumber(results.bitmap_resolution, 16, 2048), + color: results.background_color, + name: results.bitmap_name, + folder: results.bitmap_folder, + particle: 'auto', + entity_template: results.bitmap_doTemplate, + compress: results.bitmap_compressTemplate, + power: results.bitmap_power + }) dialog.hide() } - }) - dialog.show() - $('#bitmap_compressTemplate').parent().hide() + }).show() + $('#bitmap_compressTemplate, #bitmap_power').parent().hide() $('.dialog#add_bitmap input#bitmap_doTemplate').click(function() { var checked = $('.dialog#add_bitmap input#bitmap_doTemplate').is(':checked') - $('#bitmap_compressTemplate').parent()[ checked ? 'show' : 'hide' ]() + $('#bitmap_compressTemplate, #bitmap_power').parent()[ checked ? 'show' : 'hide' ]() if (Painter.background_color.get().toHex8() === 'ffffffff') { Painter.background_color.set('#00000000') } }) } - testSetup() { - Painter.addBitmap() - main_uv.setFace('up') - addCube().extend({to:[16,1,16]}) - elements[0].faces.up.uv = [0,0,16,16] - textures[0].apply() - Canvas.updateSelected() - updateSelection() - } - addBitmapFromDialog() { - var color = Painter.background_color.get() - - Painter.addBitmap({ - res: limitNumber(parseInt($('.dialog#add_bitmap input#bitmap_resolution').val()), 16, 2048), - color: color, - name: $('.dialog#add_bitmap input#bitmap_name').val(), - folder: $('.dialog#add_bitmap input#bitmap_folder').val(), - particle: 'auto', - entity_template: $('.dialog#add_bitmap input#bitmap_doTemplate').is(':checked'), - compress: $('.dialog#add_bitmap input#bitmap_compressTemplate').is(':checked') - }) - } addBitmap(options, after) { if (typeof options !== 'object') { options = {} @@ -622,11 +604,15 @@ class BBPainter { Blockbench.showMessage('No valid cubes', 'center') return; } - /* - TEMPLATE MENU - condensed - use old texture - */ + + function getNextPower(num, min) { + var i = min ? min : 2 + while (i < num && i < 4000) { + i *= 2 + } + return i; + } + if (options.compress) { var fill_map = {} @@ -726,20 +712,16 @@ class BBPainter { extend_y += max_height }) } - - //Size - //function getNextPower(num, min) { - // var i = min ? min : 2 - // while (i < num && i < 4000) { - // i *= 2 - // } - // return i; - //} + var max_size = Math.max(extend_x, extend_y) - max_size = Math.ceil(max_size/16)*16 + if (options.power) { + max_size = getNextPower(max_size, 16); + } else { + max_size = Math.ceil(max_size/16)*16; + } if (background_color.getAlpha() != 0) { - background_color = background_color.toInteger() + background_color = background_color.toRgbString() } var canvas = document.createElement('canvas') canvas.width = canvas.height = max_size*res_multiple; @@ -747,8 +729,9 @@ class BBPainter { ctx.imageSmoothingEnabled = false; - function drawTemplateRectangle(border_color, color, coords) { - if (typeof background_color === 'number') { + function drawTemplateRectangle(border_color, color, face, coords) { + //if (!Blockbench.entity_mode && face && face.texture === null) return; + if (typeof background_color === 'string') { border_color = background_color color = undefined } @@ -840,7 +823,7 @@ class BBPainter { if (!t.obj.faces[face].texture || !drawTexture(t.obj.faces[face], d.place(t)) ) { - drawTemplateRectangle(d.c1, d.c2, d.place(t)) + drawTemplateRectangle(d.c1, d.c2, t.obj.faces[face], d.place(t)) } } obj.uv_offset[0] = t.posx @@ -880,7 +863,20 @@ class BBPainter { } } } -var Painter = new BBPainter() +const Painter = new BBPainter() + +const ColorPanel = { + set: function(color) { + var value = new tinycolor(color) + $('#main_colorpicker').spectrum('set', value.toHex8String()) + $('#main_colorpicker_preview > div').css('background-color', value.toRgbString()) + return this; + }, + get: function() { + var value = $('#main_colorpicker').spectrum('get'); + return value; + } +} BARS.defineActions(function() { @@ -980,11 +976,6 @@ BARS.defineActions(function() { } }) - new ColorPicker({ - id: 'brush_color', - condition: () => (Toolbox && ['brush_tool', 'color_picker', 'fill_tool'].includes(Toolbox.selected.id)), - palette: true - }) new BarSelect({ id: 'brush_mode', condition: () => Toolbox && (Toolbox.selected.id === 'brush_tool' || Toolbox.selected.id === 'eraser'), diff --git a/js/plugin_loader.js b/js/plugin_loader.js index d79b0202..ebcb7baa 100644 --- a/js/plugin_loader.js +++ b/js/plugin_loader.js @@ -79,6 +79,9 @@ class Plugin { } download(first) { var scope = this; + if (first) { + Blockbench.showQuickMessage(tl('message.install_plugin', [scope.title]), 1400) + } if (!isApp) { scope.install(first) return this; @@ -249,14 +252,18 @@ function loadInstalledPlugins() { }) } if (Plugins.installed.length > 0) { + var loaded = [] Plugins.installed.forEach(function(id) { - if (id.substr(-3) === '.js') { + if (id && id.substr(-3) === '.js') { //Dev Plugins var plugin = new Plugin().loadFromFile({path: id}, true) + loaded.push(pathToName(id)) + } else if (id) { + loaded.push(id) } }) - console.log('Loaded '+Plugins.installed.length+' plugin'+pluralS(Plugins.installed.length)) + console.log(`Loaded ${loaded.length} plugin${pluralS(loaded.length)}`, loaded) } Plugins.Vue = new Vue({ diff --git a/js/preview.js b/js/preview.js index d3d2d834..5390050f 100644 --- a/js/preview.js +++ b/js/preview.js @@ -649,6 +649,9 @@ class Preview { } } fullscreen() { + if (quad_previews.current) { + quad_previews.current.controls.stopMovement() + } quad_previews.current = this; quad_previews.enabled = false; $('#preview').empty() @@ -1356,14 +1359,20 @@ class CanvasController { }) } updateRenderSides() { + var side = Blockbench.entity_mode ? 2 : 0; + if (display_mode) { + if (['thirdperson_righthand', 'thirdperson_lefthand', 'head'].includes(display_slot)) { + side = 2; + } + } textures.forEach(function(t) { var mat = Canvas.materials[t.uuid] if (mat) { - mat.side = (display_mode || Blockbench.entity_mode) ? 2 : 0 + mat.side = side } }) emptyMaterials.forEach(function(mat) { - mat.side = (display_mode || Blockbench.entity_mode) ? 2 : 0 + mat.side = side }) } //Selection updaters @@ -1667,8 +1676,6 @@ class CanvasController { obj.faces[f.face].uv[2] = uv[2] obj.faces[f.face].uv[3] = uv[3] - var do_cl = Math.random()<0.001 - //Fight Bleeding for (var si = 0; si < 2; si++) { let margin = 16/(si?Project.texture_height:Project.texture_width)/16; @@ -1823,36 +1830,30 @@ BARS.defineActions(function() { icon: 'local_movies', category: 'view', click: function () { - var lines = [ - {label: 'dialog.create_gif.length', node: '<input class="dark_bordered half" type="number" value="10" step="0.25" id="gif_length">'}, - {label: 'dialog.create_gif.fps', node: '<input class="dark_bordered half" type="number" value="10" id="gif_fps">'}, - {label: 'dialog.create_gif.compression', node: '<input class="dark_bordered half" type="number" value="4" id="gif_quality">'}, - ] - if (Animator.open) { - lines.push({label: 'dialog.create_gif.play', node: '<input type="checkbox" id="gif_play_animation">'}) - } - var dialog = new Dialog({ + new Dialog({ id: 'create_gif', title: tl('dialog.create_gif.title'), draggable: true, - lines: lines, - onConfirm: function() { - var jq = $(dialog.object) - var length = parseFloat( jq.find('#gif_length').val() ) - var fps = parseInt( jq.find('#gif_fps').val() ) - var quality = parseInt( jq.find('#gif_quality').val() ) - if (jq.find('#gif_play_animation').is(':checked')) { + form: { + length: {label: 'dialog.create_gif.length', type: 'number', value: 10, step: 0.25}, + fps: {label: 'dialog.create_gif.fps', type: 'number', value: 10}, + quality:{label: 'dialog.create_gif.compression', type: 'number', value: 4}, + 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) { + if (formData.play) { Timeline.start() } Screencam.createGif({ - length: limitNumber(length, 0.1, 240)*1000, - fps: limitNumber(fps, 0.5, 30), - quality: limitNumber(quality, 0, 30), + length: limitNumber(formData.length, 0.1, 240)*1000, + fps: limitNumber(formData.fps, 0.5, 30), + quality: limitNumber(formData.quality, 0, 30), + turnspeed: formData.turn, }, Screencam.returnScreenshot) - dialog.hide() + this.hide() } - }) - dialog.show() + }).show() } }) new Action({ diff --git a/js/settings.js b/js/settings.js index 17d0b77a..aaffb0aa 100644 --- a/js/settings.js +++ b/js/settings.js @@ -14,12 +14,13 @@ function settingSetup() { origin_size: {category: 'preview', value: 10, type: 'number'}, control_size: {category: 'preview', value: 10, type: 'number'}, //focal_length: {category: 'preview', value: 70, type: 'number'}, - display_skin: {category: 'preview', value: false, type: 'click', condition: isApp, icon: 'icon-player', click: function() { changeDisplaySkin() }}, seethrough_outline: {category: 'preview', value: false}, brightness: {category: 'preview', value: 50, type: 'number'}, shading: {category: 'preview', value: true}, transparency: {category: 'preview', value: true}, + outliner_colors: {category: 'preview', value: true}, texture_fps: {category: 'preview', value: 2, type: 'number'}, + display_skin: {category: 'preview', value: false, type: 'click', condition: isApp, icon: 'icon-player', click: function() { changeDisplaySkin() }}, //Edit undo_limit: {category: 'edit', value: 128, type: 'number'}, restricted_canvas: {category: 'edit', value: true}, @@ -51,7 +52,8 @@ function settingSetup() { minifiedout: {category: 'export', value: false}, export_groups: {category: 'export', value: true}, obj_textures: {category: 'export', value: true}, - credit: {category: 'export', value: 'Made with Blockbench', type: 'text'} + sketchfab_token:{category: 'export', value: '', type: 'text'}, + credit: {category: 'export', value: 'Made with Blockbench', type: 'text'}, } if (localStorage.getItem('settings') != null) { @@ -260,19 +262,6 @@ function saveSettings(force_update) { } Blockbench.dispatchEvent('update_settings') } -function saveProjectSettings() { - if (Blockbench.entity_mode) { - main_uv.setGrid() - if (uv_dialog.editors) { - uv_dialog.editors.single.setGrid() - } - if (entityMode.old_res.x !== Project.texture_width || entityMode.old_res.y !== Project.texture_height) { - entityMode.setResolution() - Undo.finishEdit('changed resolution') - } - } - hideDialog() -} function toggleSetting(setting) { if (settings[setting].value === true) { settings[setting].value = false diff --git a/js/textures.js b/js/textures.js index 7e6eae60..568b85c4 100644 --- a/js/textures.js +++ b/js/textures.js @@ -1,25 +1,28 @@ //Textures class Texture { - constructor(data) { - this.path = '' + constructor(data, uuid) { + var scope = this; + //Info + this.id = ''; this.name = '' this.folder = ''; this.namespace = ''; - this.id = ''; - this.source = '' + this.path = '' this.particle = false + //meta + this.source = '' this.selected = false - this.error = false; - this.ratio = 1 this.show_icon = true - this.average_color = {r:0, g:0, b:0} this.dark_box = false + this.error = 0; + //Data + this.ratio = 1 this.img = 0; this.res = 0; - this.mode = 'link' //link, bitmap (internally used) this.saved = true - if (!isApp) this.mode = 'bitmap' - this.uuid = guid() + + this.mode = isApp ? 'link' : 'bitmap'; + this.uuid = uuid || guid() if (typeof data === 'object') { this.extend(data) @@ -39,16 +42,129 @@ class Texture { i++; } else { this.id = i.toString(); - return; + break; } } } + //Setup Img/Mat + var img = this.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 + + var mat = new THREE.MeshLambertMaterial({ + color: 0xffffff, + map: tex, + transparent: settings.transparency.value, + side: display_mode || Blockbench.entity_mode ? 2 : 0, + alphaTest: 0.2 + }); + Canvas.materials[this.uuid] = mat + + this.img.onload = function() { + if (!this.src) return; + this.tex.needsUpdate = true; + scope.res = img.naturalWidth; + scope.ratio = img.naturalWidth / img.naturalHeight; + + if (scope.isDefault) { + console.log('Successfully loaded '+scope.name+' from default pack') + } + + var average_color = getAverageRGB(this) + scope.dark_box = (average_color.r + average_color.g + average_color.b) >= 383 + + //Width / Animation + if (img.naturalWidth !== img.naturalHeight && Blockbench.entity_mode === false) { + if (img.naturalHeight % img.naturalWidth !== 0) { + scope.error = 2; + Blockbench.showQuickMessage('message.square_textures') + } + } + + + + + //-------------------------------------------------------------------------- + + if (Blockbench.entity_mode && textures.indexOf(scope) === 0) { + if (!scope.keep_size) { + var size = { + pw: Project.texture_width, + ph: Project.texture_height, + nw: img.naturalWidth, + nh: img.naturalHeight + } + if (false && (scope.old_width != size.nw || scope.old_height != size.nh) && (size.pw != size.nw || size.ph != size.nh)) { + Blockbench.showMessageBox({ + translateKey: 'update_res', + icon: 'photo_size_select_small', + buttons: [tl('message.update_res.update'), tl('dialog.cancel')], + confirm: 0, + cancel: 1 + }, function(result) { + if (result === 0) { + var lockUV = ( // EG. Texture Optimierung > Modulo geht nicht in allen Bereichen auf + (size.pw%size.nw || size.ph%size.nh) && + (size.nw%size.pw || size.nh%size.ph) + ) + entityMode.setResolution(img.naturalWidth, img.naturalHeight, lockUV) + if (selected.length) { + main_uv.loadData() + main_uv.setGrid() + } + } + }) + } + scope.old_width = img.naturalWidth + scope.old_height = img.naturalHeight + } + } + + //-------------------------------------------------------------------------- + + + + + if ($('.dialog#texture_edit:visible').length > 0 && scope.selected === true) { + scope.openMenu() + } + TextureAnimator.updateButton() + Canvas.updateAllFaces(scope) + if (typeof scope.load_callback === 'function') { + scope.load_callback(scope); + delete scope.load_callback; + } + } + this.img.onerror = function() { + if (isApp && + !scope.isDefault && + scope.mode !== 'bitmap' && + scope.fromDefaultPack() + ) { + return true; + } else { + scope.loadEmpty() + } + } } get frameCount() { - if (this.ratio !== 1) { + if (1/this.ratio % 1 === 0) { return 1/this.ratio } } + getErrorMessage() { + switch (this.error) { + case 0: return ''; break; + case 1: return tl('texture.error.file'); break; + case 1: return tl('texture.error.invalid'); break; + case 2: return tl('texture.error.ratio'); break; + case 3: return tl('texture.error.parent'); break; + } + } getUndoCopy(bitmap) { var copy = { path: this.path, @@ -80,126 +196,17 @@ class Texture { Merge.boolean(this, data, 'saved') if (this.mode === 'bitmap') { Merge.string(this, data, 'source') + } else if (data.path) { + this.source = this.path + '?' + tex_version; } return this; } //Loading - load(isDefault, reloading, cb) { - var scope = this; - - if (Painter.current.texture === this) { - Painter.current = {} - } - this.error = false; - this.show_icon = true - var img = this.img = new Image() - - if (Canvas.materials[scope.uuid] !== undefined) { - Canvas.materials[scope.uuid].dispose() - } - function onerror() { - if (isApp && - !(isDefault || scope.isDefault) && - scope.mode !== 'bitmap' && - scope.fromDefaultPack() - ) { - return true; - } else { - scope.img.src = 'assets/missing.png' - scope.error = true; - scope.show_icon = false - console.log('Error loading '+scope.source) - } - } - if (isApp && this.mode === 'link' && !fs.existsSync(this.source.replace(/\?\d+$/, ''))) { - if (onerror()) { - return - } - } else { - img.src = this.source - img.onerror = onerror - } - - 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; - scope.res = img.naturalWidth; - scope.ratio = img.naturalWidth / img.naturalHeight; - - if (isDefault) { - console.log('Successfully loaded '+scope.name+' from default pack') - } - - scope.average_color = getAverageRGB(this) - scope.dark_box = (scope.average_color.r + scope.average_color.g + scope.average_color.b) >= 383 - - //Width / Animation - if (img.naturalWidth !== img.naturalHeight && Blockbench.entity_mode === false) { - if (img.naturalHeight % img.naturalWidth === 0) { - Canvas.updateAllUVs() - BARS.updateConditions() - } else { - scope.error = true; - Blockbench.showQuickMessage('message.square_textures') - } - } - if (Blockbench.entity_mode && textures.indexOf(scope) === 0) { - if (!scope.keep_size) { - var size = { - pw: Project.texture_width, - ph: Project.texture_height, - nw: img.naturalWidth, - nh: img.naturalHeight - } - if ((scope.old_width != size.nw || scope.old_height != size.nh) && (size.pw != size.nw || size.ph != size.nh)) { - Blockbench.showMessageBox({ - translateKey: 'update_res', - icon: 'photo_size_select_small', - buttons: [tl('message.update_res.update'), tl('dialog.cancel')], - confirm: 0, - cancel: 1 - }, function(result) { - if (result === 0) { - var lockUV = ( // EG. Texture Optimierung > Modulo geht nicht in allen Bereichen auf - (size.pw%size.nw || size.ph%size.nh) && - (size.nw%size.pw || size.nh%size.ph) - ) - entityMode.setResolution(img.naturalWidth, img.naturalHeight, lockUV) - if (selected.length) { - main_uv.loadData() - main_uv.setGrid() - } - } - }) - } - scope.old_width = img.naturalWidth - scope.old_height = img.naturalHeight - } - } - if ($('.dialog#texture_edit:visible').length >= 1 && scope.selected === true) { - scope.openMenu() - } - TextureAnimator.updateButton() - Canvas.updateAllFaces(scope) - } - if (Canvas.materials[this.uuid]) { - Canvas.materials[this.uuid].map.dispose() - Canvas.materials[this.uuid].dispose() - delete Canvas.materials[this.uuid] - } - var mat = new THREE.MeshLambertMaterial({ - color: 0xffffff, - map: tex, - transparent: settings.transparency.value, - side: display_mode || Blockbench.entity_mode ? 2 : 0, - alphaTest: 0.2 - }); - Canvas.materials[this.uuid] = mat + load(cb) { + this.error = 0; + this.show_icon = true; + this.img.src = this.source; + this.load_callback = cb; return this; } fromJavaLink(link, path_array) { @@ -283,11 +290,21 @@ class Texture { this.startWatcher() if (!isApp && Project.dataURLTextures) { - if (this.img && this.img.src) { - this.img.src = 'assets/missing.png' - } - this.error = true; - this.show_icon = false + this.loadEmpty() + + } else if (EditSession.active) { + this.load(() => { + var before = {textures: {}} + before.textures[scope.uuid] = true; + this.edit() + var post = new Undo.save({textures: [this]}) + EditSession.sendEdit({ + before: before, + post: post, + action: 'loaded_texture', + save_history: false + }) + }) } else { this.load() } @@ -305,7 +322,7 @@ class Texture { if (Blockbench.entity_mode) { var path = findEntityTexture(Project.parent, 'raw') if (path) { - this.isDefault = true + this.isDefault = true; path = settings.default_path.value + osfs + path if (fs.existsSync(path + '.png')) { @@ -320,15 +337,23 @@ class Texture { } delete this.isDefault } - } else { - var path = settings.default_path.value + osfs + this.folder.replace(/\//g, osfs) + osfs + this.name + } else if (this.name && this.name.includes('.')) { + var folder = this.folder.replace(/\//g, osfs); + var path = settings.default_path.value + osfs + (folder ? (folder+osfs) : '') + this.name if (fs.existsSync(path)) { + this.isDefault = true; this.fromPath(path) return true; } } } } + loadEmpty(error_id) { + this.img.src = 'assets/missing.png' + this.error = error_id||1; + this.show_icon = false; + return this; + } updateSource(dataUrl) { this.source = dataUrl this.img.src = dataUrl @@ -406,11 +431,9 @@ class Texture { } //this.source = this.path + '?' + tex_version; this.source = this.source.replace(/\?\d+$/, '?' + tex_version) - this.load(undefined, true) - if (single) { - main_uv.loadData() - loadTextureDraggable() - } + this.load(undefined) + TickUpdates.main_uv = true; + TickUpdates.texture_list = true; } reloadTexture() { this.refresh(true) @@ -476,7 +499,7 @@ class Texture { return this; } add(undo) { - if (undo !== false) { + if (undo) { Undo.initEdit({textures: []}) } var scope = this @@ -503,7 +526,7 @@ class Texture { } }) } - if (undo === true) { + if (undo) { Undo.finishEdit('add_texture', {textures: [this]}) } return this; @@ -567,7 +590,11 @@ class Texture { } //Interface openFolder() { - if (!isApp || !this.path) return; + if (!isApp || !this.path) return this; + if (!fs.existsSync(this.path)) { + Blockbench.showQuickMessage('texture.error.file') + return this; + } shell.showItemInFolder(this.path) return this; } @@ -627,13 +654,8 @@ class Texture { } } //Export - javaTextureLink(backup) { - if (backup) { - return this.source; - } - + javaTextureLink() { var link = this.name.replace(/\.png$/, '') - if (this.folder) { link = this.folder + '/' + link } @@ -690,7 +712,7 @@ class Texture { } return this; } - toBitmap(cb) { + getBase64() { var scope = this; if (isApp && scope.mode === 'link') { var canvas = document.createElement('canvas') @@ -698,22 +720,22 @@ class Texture { canvas.height = scope.img.naturalHeight; var ctx = canvas.getContext('2d'); ctx.drawImage(scope.img, 0, 0) - scope.mode = 'bitmap' - scope.saved = false - scope.source = canvas.toDataURL('image/png') - cb() + var dataUrl = canvas.toDataURL('image/png') + } else { + var dataUrl = scope.source } + return dataUrl.replace('data:image/png;base64,', '') } edit(cb, options) { var scope = this; - if (typeof options !== 'object') { - options = {} - } + if (!options) options = false; + if (scope.mode === 'link') { - scope.toBitmap(function() { - Painter.edit(scope, cb, options) - }) - } else { + scope.source = 'data:image/png;base64,' + scope.getBase64() + scope.mode = 'bitmap' + scope.saved = false + } + if (cb) { Painter.edit(scope, cb, options) } scope.saved = false; @@ -852,16 +874,26 @@ function saveTextures() { } function saveTextureMenu() { hideDialog() - Undo.initEdit({textures}) var tex = textures.selected - tex.name = $('#texture_edit input#te_name').val() - tex.id = $('#texture_edit input#te_variable').val() - tex.folder = $('#texture_edit input#te_folder').val() - tex.namespace = $('#texture_edit input#te_namespace').val() - $('#texture_edit #change_file_button').unbind('click') - $('#texture_edit #file_upload').unbind('input') - Undo.finishEdit('texture_edit') + var name = $('#texture_edit input#te_name').val(), + id = $('#texture_edit input#te_variable').val(), + folder = $('#texture_edit input#te_folder').val(), + namespace = $('#texture_edit input#te_namespace').val(); + + if (!( + tex.name === name && + tex.id === id && + tex.folder === folder && + tex.namespace === namespace) + ) { + Undo.initEdit({textures}) + tex.name = name; + tex.id = id; + tex.folder = folder; + tex.namespace = namespace; + Undo.finishEdit('texture_edit') + } } function loadTextureDraggable() { Vue.nextTick(function() { @@ -894,9 +926,11 @@ function loadTextureDraggable() { if (data.cube && data.face) { var tex = textures.findInArray('uuid', ui.helper.attr('texid')); var cubes_list = data.cube.selected ? selected : [data.cube]; - Undo.initEdit({}) + Undo.initEdit({cubes: cubes_list}) if (tex) { - data.cube.applyTexture(tex, [data.face]) + cubes_list.forEach(cube => { + cube.applyTexture(tex, [data.face]) + }) } Undo.finishEdit('apply texture') } diff --git a/js/transform.js b/js/transform.js index 3c49eab5..15157d25 100644 --- a/js/transform.js +++ b/js/transform.js @@ -287,12 +287,11 @@ function scaleAll(save, size) { if (size === undefined) { size = $('#model_scale_label').val() } - var origin = [8, 8, 8] - if (Blockbench.entity_mode) { - origin = [0, 0, 0] - } else if (selected_group) { - origin = selected_group.origin - } + var origin = [ + parseFloat($('#scaling_origin_x').val())||0, + parseFloat($('#scaling_origin_y').val())||0, + parseFloat($('#scaling_origin_z').val())||0, + ] var clip = false selected.forEach(function(obj) { obj.autouv = 0; @@ -735,6 +734,9 @@ BARS.defineActions(function() { } selected_group.origin[axis] += diff Canvas.updatePositions() + if (Blockbench.entity_mode) { + Canvas.updateAllBones() + } return; } selected.forEach(function(obj, i) { @@ -821,6 +823,11 @@ BARS.defineActions(function() { }, 'group', true) } showDialog('scaling') + var v = Blockbench.entity_mode ? 0 : 8; + var origin = selected_group ? selected_group.origin : [v, 0, v]; + $('#scaling_origin_x').val(origin[0]) + $('#scaling_origin_y').val(origin[1]) + $('#scaling_origin_z').val(origin[2]) scaleAll(false, 1) } }) diff --git a/js/tree.vue.js b/js/tree.vue.js index 002b96e0..7eb9d6e3 100644 --- a/js/tree.vue.js +++ b/js/tree.vue.js @@ -14,7 +14,7 @@ '<i v-if="node.children && node.children.length > 0" v-on:click="toggle(node)" class="fa icon-open-state" :class=\'{"fa-caret-right": !node.isOpen, "fa-caret-down": node.isOpen}\'></i>' + '<i v-else class="outliner_opener_placeholder"></i>' + //Main - '<i v-if="showIcon(node)" :class="nodeClass(node)"></i>' + + '<i :class="node.icon + (settings.outliner_colors.value ? \' ec_\'+node.color : \'\')"></i>' + '<input type="text" class="cube_name" v-model="node.name" disabled>' + '<a v-for="btn in node.buttons" class="ml5" href="javascript:" :title="btn.title" v-on:click.stop="btnClick(btn, node)" v-bind:class="{advanced_option: btn.advanced_option}">' + '<i v-if="node.isIconEnabled(btn) === true" :class="btn.icon"></i>' + @@ -33,9 +33,6 @@ } }, methods: { - showIcon: function (node) { - return node.icon || node.openedIcon || node.closedIcon; - }, nodeClass: function (node) { if (node.isOpen) { return node.openedIcon || node.icon; diff --git a/js/undo.js b/js/undo.js index c248b71b..315b7cae 100644 --- a/js/undo.js +++ b/js/undo.js @@ -37,6 +37,9 @@ var Undo = { Prop.project_saved = false; } Blockbench.dispatchEvent('finished_edit', {aspects}) + if (EditSession.active) { + EditSession.sendEdit(entry) + } }, cancelEdit: function() { if (!Undo.current_save) return; @@ -44,7 +47,7 @@ var Undo = { Undo.loadSave(Undo.current_save, new Undo.save(Undo.current_save.aspects)) delete Undo.current_save; }, - undo: function() { + undo: function(remote) { if (Undo.history.length <= 0 || Undo.index < 1) return; Prop.project_saved = false; @@ -52,10 +55,13 @@ var Undo = { var entry = Undo.history[Undo.index] Undo.loadSave(entry.before, entry.post) + if (EditSession.active && remote !== true) { + EditSession.sendAll('command', 'undo') + } console.log('Undo: '+entry.action) Blockbench.dispatchEvent('undo', {entry}) }, - redo: function() { + redo: function(remote) { if (Undo.history.length <= 0) return; if (Undo.index >= Undo.history.length) { return; @@ -65,9 +71,26 @@ var Undo = { var entry = Undo.history[Undo.index-1] Undo.loadSave(entry.post, entry.before) + if (EditSession.active && remote !== true) { + EditSession.sendAll('command', 'redo') + } console.log('Redo: '+entry.action) Blockbench.dispatchEvent('redo', {entry}) }, + remoteEdit: function(entry) { + Undo.loadSave(entry.post, entry.before, 'session') + + if (entry.save_history !== false) { + delete Undo.current_save; + Undo.history.push(entry) + if (Undo.history.length > settings.undo_limit.value) { + Undo.history.shift() + } + Undo.index = Undo.history.length + Prop.project_saved = false; + Blockbench.dispatchEvent('finished_edit', {remote: true}) + } + }, getItemByUUID: function(list, uuid) { if (!list || typeof list !== 'object' || !list.length) {return false;} var i = 0; @@ -96,16 +119,15 @@ var Undo = { if (aspects.cubes) { this.cubes = {} aspects.cubes.forEach(function(obj) { + var copy = new Cube(obj) if (aspects.uv_only) { - var copy = new Cube(obj) copy = { uv_offset: copy.uv_offset, faces: copy.faces, } - } else { - var copy = new Cube(obj) } copy.uuid = obj.uuid + delete copy.parent; scope.cubes[obj.uuid] = copy }) } @@ -138,8 +160,11 @@ var Undo = { } } - if (aspects.animation) { - this.animation = aspects.animation ? aspects.animation.undoCopy() : null; + if (aspects.animations) { + this.animations = {} + aspects.animations.forEach(a => { + scope.animations[a.uuid] = a.undoCopy(); + }) } if (aspects.keyframes && Animator.selected && Animator.selected.getBoneAnimator()) { this.keyframes = { @@ -162,7 +187,8 @@ var Undo = { }) } }, - loadSave: function(save, reference) { + loadSave: function(save, reference, mode) { + var is_session = mode === 'session'; if (save.cubes) { for (var uuid in save.cubes) { if (save.cubes.hasOwnProperty(uuid)) { @@ -185,7 +211,7 @@ var Undo = { if (reference.cubes.hasOwnProperty(uuid) && !save.cubes.hasOwnProperty(uuid)) { var obj = elements.findInArray('uuid', uuid) if (obj) { - obj.remove(false) + obj.remove() } } } @@ -196,12 +222,23 @@ var Undo = { if (save.outliner) { selected_group = undefined parseGroups(save.outliner) + if (is_session) { + function iterate(arr) { + arr.forEach((obj) => { + delete obj.isOpen; + if (obj.children) { + iterate(obj.children) + } + }) + } + iterate(save.outliner) + } if (Blockbench.entity_mode) { Canvas.updateAllPositions() } } - if (save.selection_group) { + if (save.selection_group && !is_session) { selected_group = undefined var sel_group = TreeElements.findRecursive('uuid', save.selection_group) if (sel_group) { @@ -209,7 +246,7 @@ var Undo = { } } - if (save.selection) { + if (save.selection && !is_session) { selected.length = 0; elements.forEach(function(obj) { if (save.selection.includes(obj.uuid)) { @@ -221,6 +258,9 @@ var Undo = { if (save.group) { var group = TreeElements.findRecursive('uuid', save.group.uuid) if (group) { + if (is_session) { + delete save.group.isOpen; + } group.extend(save.group) if (Blockbench.entity_mode) { group.forEachChild(function(obj) { @@ -238,10 +278,17 @@ var Undo = { if (reference.textures[uuid]) { var tex = Undo.getItemByUUID(textures, uuid) if (tex) { + var require_reload = tex.mode !== save.textures[uuid].mode; tex.extend(save.textures[uuid]).updateMaterial() + if (require_reload || reference.textures[uuid] === true) { + tex.load() + } else { + tex.updateMaterial() + } } } else { - new Texture(save.textures[uuid]).load().add(false) + var tex = new Texture(save.textures[uuid], uuid) + tex.load().add(false) } } for (var uuid in reference.textures) { @@ -265,81 +312,91 @@ var Undo = { Project.texture_height = save.resolution.height } - if (save.animation) { + if (save.animations) { + for (var uuid in save.animations) { - var animation = Animator.animations.findInArray('uuid', save.animation) - if (!animation) { - animation = new Animation() - } - Animation.extend(save.animation) - - } else if (reference.animation) { - //remove - var animation = Animator.animations.findInArray('uuid', reference.animation.uuid) - if (animation.remove) { - animation.remove() - } - } - - if (save.keyframes && Animator.selected) { - var animation = false; - if (Animator.selected.uuid !== save.keyframes.animation) { - animation = Animator.animations.findInArray('uuid', save.keyframes.animation) - if (animation.select) { + var animation = reference.animations[uuid] ? Undo.getItemByUUID(Animator.animations, uuid) : null; + if (!animation) { + animation = new Animation() + animation.uuid = uuid + } + animation.extend(save.animations[uuid]).add(false) + if (save.animations[uuid].selected) { animation.select() } } - - var bone = Animator.selected.getBoneAnimator(); - if (!bone || bone.uuid !== save.keyframes.bone) { - for (var uuid in Animator.selected.bones) { - if (uuid === save.keyframes.bone) { - bone = Animator.selected.bones[uuid] - bone.select() + for (var uuid in reference.animations) { + if (!save.animations[uuid]) { + var animation = Undo.getItemByUUID(Animator.animations, uuid) + if (animation) { + animation.remove(false) } } } + } - - function getKeyframe(uuid) { - var i = 0; - while (i < Timeline.keyframes.length) { - if (Timeline.keyframes[i].uuid === uuid) { - return Timeline.keyframes[i]; - } - i++; + if (save.keyframes) { + var animation = Animator.selected; + if (!animation || animation.uuid !== save.keyframes.animation) { + animation = Animator.animations.findInArray('uuid', save.keyframes.animation) + if (animation.select && Animator.open && is_session) { + animation.select() } } - var added = 0; - for (var uuid in save.keyframes) { - if (uuid.length === 36 && save.keyframes.hasOwnProperty(uuid)) { - var data = save.keyframes[uuid] - var kf = getKeyframe(uuid) - if (kf) { - kf.extend(data) - } else { - kf = new Keyframe(data) - kf.parent = bone; - kf.uuid = uuid; - Timeline.keyframes.push(kf) - added++; + if (animation) { + var bone = Animator.selected.getBoneAnimator(); + if (!bone || bone.uuid !== save.keyframes.bone) { + for (var uuid in Animator.selected.bones) { + if (uuid === save.keyframes.bone) { + bone = Animator.selected.bones[uuid] + if (bone.select && Animator.open && is_session) { + bone.select() + } + } } } - } + if (bone) { - for (var uuid in reference.keyframes) { - if (uuid.length === 36 && reference.keyframes.hasOwnProperty(uuid) && !save.keyframes.hasOwnProperty(uuid)) { - var kf = getKeyframe(uuid) - if (kf) { - kf.remove() + function getKeyframe(uuid) { + var i = 0; + while (i < Timeline.keyframes.length) { + if (Timeline.keyframes[i].uuid === uuid) { + return Timeline.keyframes[i]; + } + i++; + } } + var added = 0; + for (var uuid in save.keyframes) { + if (uuid.length === 36 && save.keyframes.hasOwnProperty(uuid)) { + var data = save.keyframes[uuid] + var kf = getKeyframe(uuid) + if (kf) { + kf.extend(data) + } else { + kf = new Keyframe(data) + kf.parent = bone; + kf.uuid = uuid; + Timeline.keyframes.push(kf) + added++; + } + } + } + for (var uuid in reference.keyframes) { + if (uuid.length === 36 && reference.keyframes.hasOwnProperty(uuid) && !save.keyframes.hasOwnProperty(uuid)) { + var kf = getKeyframe(uuid) + if (kf) { + kf.remove() + } + } + } + if (added) { + Vue.nextTick(Timeline.update) + } + updateKeyframeSelection() + Animator.preview() } } - if (added) { - Vue.nextTick(Timeline.update) - } - updateKeyframeSelection() - Animator.preview() } if (save.display_slots) { diff --git a/js/util.js b/js/util.js index 7e1f6fe2..b3d43d82 100644 --- a/js/util.js +++ b/js/util.js @@ -275,6 +275,9 @@ Array.prototype.allEqual = function(s) { } return true; } +Array.prototype.random = function() { + return this[Math.floor(Math.random()*this.length)] +} //Object Object.defineProperty(Array.prototype, "equals", {enumerable: false}); diff --git a/js/uv.js b/js/uv.js index c7496df9..ef4a4273 100644 --- a/js/uv.js +++ b/js/uv.js @@ -67,6 +67,7 @@ class UVEditor { constructor(id, headline, toolbar) { this.face = 'north'; this.size = 320; + this.zoom = 1; this.grid = 16; this.id = id this.autoGrid = true; @@ -95,12 +96,15 @@ class UVEditor { uv_dialog.select(scope.id, event) }) } + this.jquery.viewport = $('<div id="uv_viewport"></div>') + this.jquery.transform_info = $('<div class="uv_transform_info"></div>') + this.jquery.main.append(this.jquery.transform_info) + this.jquery.main.append(this.jquery.viewport) + this.jquery.frame = $('<div id="uv_frame" style="background-repeat: no-repeat;"><div id="uv_size"><div class="uv_size_handle"></div></div></div>') this.jquery.size = this.jquery.frame.find('div#uv_size') - this.jquery.main.append(this.jquery.frame) - this.jquery.frame.append('<div class="uv_transform_info" title="Transform indicators"></div>') + this.jquery.viewport.append(this.jquery.frame) this.jquery.frame.css('background-repeat', 'no-repeat') - this.jquery.transform_info = this.jquery.frame.find('.uv_transform_info') if (Blockbench.browser === 'firefox') { this.jquery.frame.css('image-rendering', '-moz-crisp-edges') } @@ -222,13 +226,6 @@ class UVEditor { } - this.jquery.size.mouseenter(function() { - scope.displayMappingOverlay() - }) - this.jquery.size.mouseleave(function() { - console.trace('RM') - $(this).find('.uv_mapping_overlay').remove() - }) if (toolbar) { this.jquery.bar = $(Toolbars.main_uv.node) @@ -237,7 +234,7 @@ class UVEditor { this.jquery.bar = $('') } - + var dragging_not_clicking = false; this.jquery.size.resizable({ handles: "all", maxHeight: 320, @@ -247,34 +244,45 @@ class UVEditor { Undo.initEdit({cubes: selected, uv_only: true}) }, resize: function(event, ui) { + + + + //ui.size.width = ui.originalSize.width + (ui.size.width - ui.originalSize.width) / scope.zoom + //ui.size.height = ui.originalSize.height + (ui.size.height - ui.originalSize.height) / scope.zoom + /* + ui.size.width = ui.size.width - ui.size.width % (scope.size/scope.grid) + (scope.size/scope.grid)/2; + ui.size.height = ui.size.height - ui.size.height % (scope.size/scope.grid) + (scope.size/scope.grid)/2; + */ + //var size = main_uv.height / main_uv.grid * main_uv.zoom + scope.save() scope.displaySliders() }, stop: function(event, ui) { + dragging_not_clicking = true; Undo.finishEdit('uv_change') scope.disableAutoUV() scope.updateDragHandle(ui.position) - }, - grid: [20,20] + } }) this.jquery.size.draggable({ - containment: 'parent', start: function(event, ui) { Undo.initEdit({cubes: selected, uv_only: true}) }, drag: function( event, ui ) { - var snapTolerance = 200//$(this).draggable('option', 'snapTolerance'); - var topRemainder = ui.position.top % (scope.size/scope.grid); - var leftRemainder = ui.position.left % (scope.size/scope.grid); + var p = ui.position; + var o = ui.originalPosition + + p.left = o.left + (p.left - o.left) + p.top = o.top + (p.top - o.top) + + p.left = limitNumber(p.left, 0, scope.inner_size-scope.jquery.size.width()+1) + p.top = limitNumber(p.top, 0, scope.inner_size-scope.jquery.size.height()+1) - if (topRemainder <= snapTolerance) { - ui.position.top = ui.position.top - topRemainder; - } - - if (leftRemainder <= snapTolerance) { - ui.position.left = ui.position.left - leftRemainder; - } + p.left = p.left - p.left % (scope.inner_size/scope.grid); + p.top = p.top - p.top % (scope.inner_size/scope.grid); + scope.save() scope.displaySliders() }, @@ -304,15 +312,63 @@ class UVEditor { } }) - this.jquery.frame.contextmenu(function(event) { + this.jquery.size.mouseenter(function() { + scope.displayMappingOverlay() + }) + this.jquery.size.mouseleave(function() { + $(this).find('.uv_mapping_overlay').remove() + }) + + this.jquery.frame.click(function(event) { + if (!dragging_not_clicking) { + scope.reverseSelect(event) + } + dragging_not_clicking = false; + }) + + this.jquery.viewport.contextmenu(function(event) { scope.contextMenu() }) - this.jquery.frame.mousedown(function(event) { - if (Toolbox.selected.paintTool) { + this.jquery.viewport.mousedown(function(event) { + if (Toolbox.selected.paintTool && event.which === 1) { scope.startBrush(event) } }) + this.jquery.viewport.on('mousewheel', function(e) { + if (e.ctrlKey) { + var n = (event.deltaY < 0) ? 0.1 : -0.1; + n *= scope.zoom + var number = limitNumber(scope.zoom + n, 1.0, 4.0) + if (Math.abs(number - scope.zoom) > 0.001) { + this.scrollLeft += (scope.inner_size * n / 2) * (event.offsetX / scope.jquery.frame.width()); + this.scrollTop += (scope.inner_size * n / 2) * (event.offsetY / scope.jquery.frame.height()); + } + scope.setZoom(number) + event.preventDefault() + e.preventDefault() + return false; + } + }) + var dMWCoords = {x: 0, y: 0} + function dragMouseWheel(e) { + e.currentTarget.scrollLeft -= (e.pageX - dMWCoords.x) + e.currentTarget.scrollTop -= (e.pageY - dMWCoords.y) + dMWCoords = {x: e.pageX, y: e.pageY} + } + function dragMouseWheelStop(e) { + scope.jquery.viewport.off('mousemove', dragMouseWheel) + document.removeEventListener('mouseup', dragMouseWheelStop) + } + scope.jquery.viewport.on('mousedown', function(e) { + if (e.which === 2) { + scope.jquery.viewport.on('mousemove', dragMouseWheel) + document.addEventListener('mouseup', dragMouseWheelStop) + dMWCoords = {x: e.pageX, y: e.pageY} + e.preventDefault(); + return false; + } + }) this.setSize(this.size) return this; } @@ -331,7 +387,7 @@ class UVEditor { getBrushCoordinates(event, tex) { var scope = this; var multiplier = (Blockbench.entity_mode && tex) ? tex.res/Project.texture_width : 1 - var pixel_size = scope.size / tex.res + var pixel_size = scope.inner_size / tex.res return { x: Math.floor(event.offsetX/scope.getPixelSize()*multiplier), y: Math.floor(event.offsetY/scope.getPixelSize()*multiplier) @@ -398,12 +454,18 @@ class UVEditor { Painter.stopBrush() } //Get + get inner_size() { + return this.size*this.zoom; + } + get inner_height() { + return this.height*this.zoom; + } getPixelSize() { if (Blockbench.entity_mode) { this.grid = Project.texture_width - return this.size/this.grid + return this.inner_size/this.grid } else { - return this.size/ ( + return this.inner_size/ ( (typeof this.texture === 'object' && this.texture.res) ? this.texture.res : this.grid @@ -428,6 +490,42 @@ class UVEditor { getTexture() { return selected[0].faces[this.face].getTexture() } + reverseSelect(event) { + var scope = this; + if (!event.target.classList.contains('uv_size_handle') && !event.target.id === 'uv_frame') { + return this; + } + var matches = []; + var face_match; + var u = event.offsetX / main_uv.inner_size * 16; + var v = event.offsetY / main_uv.inner_height * 16; + elements.forEach(cube => { + for (var face in cube.faces) { + var uv = cube.faces[face].uv + if (uv && Math.isBetween(u, uv[0], uv[2]) && Math.isBetween(v, uv[1], uv[3]) && cube.faces[face].getTexture() === scope.texture) { + matches.safePush(cube) + if (!face_match) { + face_match = face + } + break; + } + } + }) + if (matches.length) { + if (!Blockbench.entity_mode) { + main_uv.setFace(face_match) + } + if (!event.ctrlKey && !event.shiftKey) { + selected.empty(); + } + matches.forEach(s => { + selected.safePush(s) + }) + updateSelection() + scope.displayMappingOverlay() + } + return this; + } forCubes(cb) { var i = 0; while (i < selected.length) { @@ -436,37 +534,63 @@ class UVEditor { } } //Set - setSize(size, cancel_load) { + setSize(input_size, cancel_load) { var old_size = this.size; - this.size = size - this.jquery.frame.width(size) - if (uv_dialog.editors !== undefined && this === uv_dialog.editors.single) { - this.jquery.main.width(size) - } + var size = input_size - (input_size % 16); + this.size = size; + this.jquery.frame.width(this.inner_size); + this.jquery.viewport.width(size+8); + this.jquery.main.width(size+8); if (Blockbench.entity_mode) { this.height = size / (Project.texture_width/Project.texture_height) - this.jquery.frame.height(this.height) + this.jquery.frame.height(this.inner_height) + this.jquery.viewport.height(this.height+8) $('.panel#textures').css('top', 133+(size / (Project.texture_width/Project.texture_height))+'px') if (old_size !== size) { this.displayAllMappingOverlays(true) } } else { this.height = size - this.jquery.frame.height(size) + this.jquery.frame.height(this.inner_size) + this.jquery.viewport.height(size+8) - this.jquery.size.resizable('option', 'maxHeight', size) - this.jquery.size.resizable('option', 'maxWidth', size) - this.jquery.size.resizable('option', 'grid', [size/this.grid, size/this.grid]) + this.jquery.size.resizable('option', 'maxHeight', this.inner_size) + this.jquery.size.resizable('option', 'maxWidth', this.inner_size) + this.jquery.size.resizable('option', 'grid', [this.inner_size/this.grid, this.inner_size/this.grid]) } for (var id in this.sliders) { - this.sliders[id].setWidth(size/(Blockbench.entity_mode?2:4)-3) + this.sliders[id].setWidth(size/(Blockbench.entity_mode?2:4)-1) } if (!cancel_load) { this.loadData() } return this; } + setZoom(zoom) { + var zoomed_size = this.size * zoom; + var size = zoomed_size - (zoomed_size % 16); + this.zoom = size/this.size + + this.jquery.frame.width(this.inner_size); + + if (Blockbench.entity_mode) { + this.jquery.frame.height(this.inner_height) + this.displayAllMappingOverlays(true) + } else { + this.jquery.frame.height(this.inner_size) + this.jquery.size.resizable('option', 'maxHeight', this.inner_size) + this.jquery.size.resizable('option', 'maxWidth', this.inner_size) + this.jquery.size.resizable('option', 'grid', [this.inner_size/this.grid, this.inner_size/this.grid]) + } + if (this.zoom > 1) { + this.jquery.viewport.css('overflow', 'scroll scroll') + } else { + this.jquery.viewport.css('overflow', 'hidden') + } + this.loadData() + return this; + } setGrid(grid, load) { if (Blockbench.entity_mode) { this.autoGrid = false; @@ -504,19 +628,14 @@ class UVEditor { return this; } setFrameColor(black) { - if (black) { - this.jquery.size.css('box-shadow', '0 0 6px black') - } else { - this.jquery.size.css('box-shadow', '0 0 6px white') - } + this.jquery.size.toggleClass('dark_frame', black === true) } setToMainSlot() { var scope = this; $('.panel#uv').append(this.jquery.main) - this.jquery.main.on('mousewheel', function() { + this.jquery.main.on('mousewheel', function(e) { - if (Blockbench.entity_mode) { - } else { + if (!Blockbench.entity_mode && !e.ctrlKey) { var faceIDs = {'north': 0, 'south': 1, 'west': 2, 'east': 3, 'up': 4, 'down': 5} var id = faceIDs[scope.face] event.deltaY > 0 ? id++ : id--; @@ -524,6 +643,7 @@ class UVEditor { if (id === -1) id = 5 $('input#'+getKeyByValue(faceIDs, id)+'_radio').prop("checked", true) scope.loadSelectedFace() + e.preventDefault() } }) this.jquery.frame.on('dblclick', function() { @@ -571,20 +691,20 @@ class UVEditor { selected.forEach(function(obj) { obj.uv_offset = [ - Math.round(scope.jquery.size.position().left / (scope.size/Project.texture_width) * 8) / 8, - Math.round(scope.jquery.size.position().top / (scope.size/Project.texture_width) * 8) / 8 + Math.round(scope.jquery.size.position().left / (scope.inner_size/Project.texture_width) * 8) / 8, + Math.round(scope.jquery.size.position().top / (scope.inner_size/Project.texture_width) * 8) / 8 ] Canvas.updateUV(obj) }) } else { var trim = v => Math.round(v*1000+0.3)/1000; - var pixelSize = this.size/16 + var pixelSize = this.inner_size/16 var left = trim( this.jquery.size.position().left / pixelSize); var top = trim( this.jquery.size.position().top / pixelSize * (Project.texture_width/Project.texture_height)); - var left2= Math.clamp(trim( (this.jquery.size.width()) / pixelSize + left), 0, 16); - var top2 = Math.clamp(trim( (this.jquery.size.height()) / pixelSize + top), 0, 16); + var left2= Math.clamp(trim( Math.round(this.jquery.size.width()) / pixelSize + left), 0, 16); + var top2 = Math.clamp(trim( Math.round(this.jquery.size.height()) / pixelSize + top), 0, 16); var uvTag = this.getUVTag() @@ -595,11 +715,7 @@ class UVEditor { top2 = [top, top = top2][0]; } var uvArr = [left, top, left2, top2] - uvArr.forEach(function(s, i) { - if (s === 15.9) { - uvArr[i] = 16 - } - }) + selected.forEach(function(obj) { obj.faces[scope.face].uv = uvArr.slice() Canvas.updateUV(obj) @@ -623,7 +739,12 @@ class UVEditor { displayTexture(face) { var tex = face.getTexture() if (!tex || typeof tex !== 'object' || tex.error) { - this.displayEmptyTexture() + this.jquery.frame.css('background-color', 'var(--color-back)').css('background-image', 'none') + this.texture = false; + this.setFrameColor() + if (this.autoGrid || Blockbench.entity_mode) { + this.setGrid(16, false) + } } else { this.setFrameColor(tex.dark_box) var css = 'url("'+tex.source.split('\\').join('\\\\').replace(/ /g, '%20')+'")' @@ -634,11 +755,15 @@ class UVEditor { this.jquery.frame.css('background-size', 'cover') } this.texture = tex; - tex.select() if (this.autoGrid || Blockbench.entity_mode) { this.setGrid(tex.res, false) } } + if (!tex || typeof tex !== 'object') { + unselectTextures() + } else { + tex.select() + } if (Blockbench.entity_mode) { this.setSize(this.size, true) } @@ -658,14 +783,6 @@ class UVEditor { this.jquery.transform_info.append('<b>'+ref.rotation+'</b>') } } - displayEmptyTexture() { - this.jquery.frame.css('background-color', 'var(--color-back)').css('background-image', 'none') - this.texture = false; - this.setFrameColor() - if (this.autoGrid) { - this.grid = 16 - } - } displayFrame() { var scope = this; if (Blockbench.entity_mode) { @@ -675,10 +792,10 @@ class UVEditor { var width = (size_tag[0] + size_tag[2])*2 width = limitNumber(width, 0, Project.texture_width) - width = width/Project.texture_width*scope.size + width = width/Project.texture_width*scope.inner_size var x = limitNumber(uvTag[0], 0, Project.texture_width) - x *= scope.size/Project.texture_width + x *= scope.inner_size/Project.texture_width this.jquery.size.width(width) this.jquery.size.css('left', x+'px') @@ -686,11 +803,11 @@ class UVEditor { var height = size_tag[2] + size_tag[1] height = limitNumber(height, 0, Project.texture_height) - height = height/Project.texture_height*scope.size + height = height/Project.texture_height*scope.inner_size height *= Project.texture_height/Project.texture_width var y = limitNumber(uvTag[1], 0, Project.texture_height) - y *= scope.size/Project.texture_height + y *= scope.inner_size/Project.texture_height y *= Project.texture_height/Project.texture_width this.jquery.size.height(height) @@ -698,7 +815,7 @@ class UVEditor { } else { var uvTag = this.getUVTag(selected[0]) - var pixels = this.size/16 + var pixels = this.inner_size/16 //X var width = limitNumber(uvTag[2]-uvTag[0], -16, 16) @@ -729,7 +846,7 @@ class UVEditor { var scope = this; var sides = this.getMappingOverlay() - $(scope.jquery.size).find('.uv_mapping_overlay').remove() + $(scope.jquery.size).find('.mapping_overlay_cube').remove() scope.jquery.size.append(sides) return this; @@ -746,8 +863,8 @@ class UVEditor { } x *= pixels; y *= pixels; - width = limitNumber(width *pixels + x, 0, scope.size) - x; - height = limitNumber(height*pixels + y, 0, scope.height)- y; + width = limitNumber(width *pixels + x, 0, scope.inner_size) - x; + height = limitNumber(height*pixels + y, 0, scope.inner_height)- y; sides.append($(`<div class="uv_mapping_overlay" style="left: ${x}px; top: ${y}px; @@ -774,7 +891,7 @@ class UVEditor { elements.forEach(cube => { var size = cube.size(undefined, true) var hash = `${cube.uv_offset[0]}_${cube.uv_offset[1]}_${size[0]}_${size[1]}_${size[2]}` - var c = scope.jquery.frame.find(`.mapping_overlay_cube:not(.${cycle})[size_hash="${hash}"]`).first() + var c = scope.jquery.frame.find(`> .mapping_overlay_cube:not(.${cycle})[size_hash="${hash}"]`).first() if (force_reload || !c.length) { var sides = scope.getMappingOverlay(cube, true) sides.addClass(cycle) @@ -819,7 +936,7 @@ class UVEditor { } contextMenu() { var scope = this; - if (Blockbench.entity_mode) return; + //if (Blockbench.entity_mode) return; this.reference_face = selected[0].faces[scope.face] this.menu.open(event, this) return this; @@ -1228,6 +1345,12 @@ class UVEditor { } } UVEditor.prototype.menu = new Menu([ + {name: 'menu.view.zoom', id: 'zoom', condition: isApp, icon: 'search', children: [ + 'zoom_in', + 'zoom_out', + 'zoom_reset' + ]}, + '_', 'copy', 'paste', /* @@ -1239,7 +1362,7 @@ class UVEditor { editor.paste(event) Undo.finishEdit('uv_paste') }},*/ - {icon: 'photo_size_select_large', name: 'menu.uv.mapping', children: function(editor) { return [ + {icon: 'photo_size_select_large', name: 'menu.uv.mapping', condition: () => !Blockbench.entity_mode, children: function(editor) { return [ {icon: editor.reference_face.enabled!==false ? 'check_box' : 'check_box_outline_blank', name: 'menu.uv.mapping.export', click: function(editor) { Undo.initEdit({cubes: selected, uv_only: true}) editor.toggleUV(event) @@ -1295,13 +1418,14 @@ class UVEditor { ]}}, { icon: (editor) => (editor.reference_face.tint ? 'check_box' : 'check_box_outline_blank'), + condition: () => !Blockbench.entity_mode, name: 'menu.uv.tint', click: function(editor) { Undo.initEdit({cubes: selected, uv_only: true}) editor.switchTint(selected[0].faces[editor.face].tint) Undo.finishEdit('face_tint') } }, - {icon: 'collections', name: 'menu.uv.texture', children: function() { + {icon: 'collections', name: 'menu.uv.texture', condition: () => !Blockbench.entity_mode, children: function() { var arr = [ {icon: 'crop_square', name: 'menu.cube.texture.blank', click: function(editor, event) { Undo.initEdit({cubes: selected}) @@ -1351,9 +1475,8 @@ const uv_dialog = { down: new UVEditor('down', true).appendTo('#uv_dialog_all') } var size = $(window).height() - 200 - size = size - (size % 16) uv_dialog.editors.single.setSize(size) - uv_dialog.editors.single.jquery.main.css('margin-left', 'auto').css('margin-right', 'auto').css('width', size+'px') + uv_dialog.editors.single.jquery.main.css('margin-left', 'auto').css('margin-right', 'auto')//.css('width', (size+10)+'px') uv_dialog.editors.up.jquery.main.css('margin-left', '276px').css('clear', 'both') uv_dialog.isSetup = true @@ -1476,7 +1599,6 @@ const uv_dialog = { BarItems.uv_grid.set(uv_dialog.editors.single.gridSelectOption) var max_size = $(window).height() - 200 - max_size = max_size - (max_size % 16) if (max_size < uv_dialog.editors.single.size ) { uv_dialog.editors.single.setSize(max_size) uv_dialog.editors.single.jquery.main.css('margin-left', 'auto').css('margin-right', 'auto').css('width', max_size+'px') @@ -1503,29 +1625,23 @@ const uv_dialog = { y: obj.height() } if (uv_dialog.single) { - var menu_gap = Blockbench.entity_mode ? 66 : 130 + var menu_gap = Blockbench.entity_mode ? 66 : 154 var editor_size = size.x size.y = (size.y - menu_gap) * (Blockbench.entity_mode ? Project.texture_width/Project.texture_height : 1) if (size.x > size.y) { editor_size = size.y } - editor_size = editor_size - (editor_size % 16) uv_dialog.editors.single.setSize(editor_size) } else { var centerUp = false if (size.x < size.y/1.2) { - //2 x 3 0.83 - 7.2 - if (size.y*1.4 > size.x) { - var editor_size = limitNumber(size.x / 2 - 20, 80, $(window).height()/3-120) - editor_size = limitNumber(editor_size, 80, (size.y-64)/3-77) - } else { - var editor_size = size.y / 3 - 96 - 48 - } + var editor_size = limitNumber(size.x / 2 - 35, 80, $(window).height()/3-120) + editor_size = limitNumber(editor_size, 80, (size.y-64)/3-120) } else { //4 x 2 - var y_margin = 150 - var editor_size = limitNumber(size.x/4-20, 16, size.y/2-y_margin) + var y_margin = 130 + var editor_size = limitNumber(size.x/4-25, 16, size.y/2-y_margin) centerUp = true } editor_size = editor_size - (editor_size % 16) @@ -1647,9 +1763,13 @@ BARS.defineActions(function() { category: 'uv', condition: () => !Blockbench.entity_mode && selected.length, min: 0, max: 270, step: 90, width: 80, - onChange: function(slider) { + onBefore: () => { Undo.initEdit({cubes: selected, uv_only: true}) + }, + onChange: function(slider) { uv_dialog.forSelection('rotate') + }, + onAfter: () => { Undo.finishEdit('uv rotate') } }) @@ -1658,6 +1778,7 @@ BARS.defineActions(function() { category: 'uv', condition: () => !Blockbench.entity_mode && selected.length, width: 60, + value: 'auto', options: { auto: true, '16': '16x16', diff --git a/js/web.js b/js/web.js index 3f9c66e7..c655802d 100644 --- a/js/web.js +++ b/js/web.js @@ -11,6 +11,9 @@ $(document).ready(function() { window.open(event.target.href, '_blank'); }); }) +/*setInterval(function() { + Prop.zoom = Math.round(devicePixelRatio*100) +}, 500)*/ function tryLoadPOSTModel() { if ($('#post_model').text() !== '') { diff --git a/lang/de.json b/lang/de.json index 41846cf9..e07f7d63 100644 --- a/lang/de.json +++ b/lang/de.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Angelpunkt auf der Z Achse verschieben", "action.brush_mode": "Pinselmodus", "action.brush_mode.desc": "Modus des Pinsels", - "action.brush_color": "Farbe", - "action.brush_color.desc": "Farbe des Pinsels", "action.slider_brush_size": "Radius", "action.slider_brush_size.desc": "Radius des Pinsels in Pixeln", "action.slider_brush_opacity": "Deckkraft", @@ -377,7 +375,7 @@ "action.export_optifine_full": "OptiFine JEM exportieren", "action.export_optifine_full.desc": "Ein vollständiges Entitymodell für OptiFine exportieren", "action.export_obj": "OBJ Modell exportieren", - "action.export_obj.desc": "Ein Wavefront OBJ Modell exportieren zur Weiterverwendung in anderen Programmen oder zum Hochladen auf Sketchfab", + "action.export_obj.desc": "Ein Wavefront OBJ Modell exportieren zum Rendern oder für Spiel-Engines", "action.save": "Speichern", "action.save.desc": "Das aktuelle Modell und die aktuellen Texturen speichern", "action.settings_window": "Einstellungen...", @@ -586,7 +584,6 @@ "panel.outliner": "Outliner", "panel.options": "Drehung", "panel.options.angle": "Winkel", - "panel.options.origin": "Angelpunkt", "uv_editor.title": "UV Bearbeiten", "uv_editor.all_faces": "Alle", "uv_editor.no_faces": "Keine", @@ -854,5 +851,53 @@ "action.export_bbmodel": "Blockbench-Projekt exportieren", "action.export_bbmodel.desc": "Exportiere ein Blockbench Projekt mit allen Elementen, Texturen und Animationen", "action.export_asset_archive": "ZIP-Ordner herunterladen", - "action.export_asset_archive.desc": "Lädt ein Archiv mit dem Modell und allen benötigten Texturen herunter" + "action.export_asset_archive.desc": "Lädt ein Archiv mit dem Modell und allen benötigten Texturen herunter", + "action.upload_sketchfab": "Teilen auf Sketchfab", + "message.sketchfab.name_or_token": "Bitte gebe deinen Sketchfab Schlüssel und einen Namen ein", + "dialog.sketchfab_uploader.title": "Modell auf Sketchfab hochladen", + "dialog.sketchfab_uploader.token": "API Schlüssel", + "dialog.sketchfab_uploader.about_token": "Der Schlüssel wird benötigt, um Blockbench mit deinem Sketchfab Account zu verbinden. Du findest ihn unter sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Name", + "dialog.sketchfab_uploader.description": "Beschreibung", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Schlüssel", + "settings.sketchfab_token.desc": "Schlüssel, der Blockbench erlaubt, Modelle auf Sketchfab hochzuladen", + "panel.color": "Farbauswahl", + "data.origin": "Angelpunkt", + "message.sketchfab.success": "Modell erfolgreich hochgeladen", + "message.sketchfab.error": "Upload auf Sketchfab fehlgeschlagen", + "settings.outliner_colors": "Outliner Farben", + "settings.outliner_colors.desc": "Elementfarben im Outliner anzeigen", + "action.upload_sketchfab.desc": "Modell auf Sketchfab hochladen", + "action.element_colors": "Elementfarben", + "action.element_colors.desc": "Elementfarben im Outliner anzeigen", + "texture.error.file": "Datei nicht gefunden", + "texture.error.invalid": "Datei fehlerhaft", + "texture.error.ratio": "Ungültiges Seitenverhältnis", + "texture.error.parent": "Textur durch Elternmodell", + "message.recover_backup.title": "Modell wiederherstellen", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/en.json b/lang/en.json index 224ab4f4..0d93f962 100644 --- a/lang/en.json +++ b/lang/en.json @@ -15,6 +15,7 @@ "data.cubes": "Cubes", "data.group": "Group", "data.texture": "Texture", + "data.origin": "Origin", "data.plugin": "Plugin", "data.preview": "Preview", "data.toolbar": "Toolbar", @@ -72,6 +73,8 @@ "message.rotation_limit.message": "Rotations are limited by Minecraft to one axis and 22.5 degree increments. Rotating on a different axis will clear all rotations on the other axes. Disable the option \"Restricted Rotation\" if you are modeling for other purposes and need free rotations.", "message.file_not_found.title": "File Not Found", "message.file_not_found.message": "Blockbench could not find the requested file. Make sure it is saved locally and not in a cloud.", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", "message.screenshot.title": "Screenshot", "message.screenshot.message": "Screenshot captured.", "message.screenshot.clipboard": "Clipboard", @@ -145,6 +148,7 @@ "message.load_plugin_app": "Do you want to allow this plugin to make changes to your PC? Only load plugins from people you trust.", "message.load_plugin_web": "Do you want to load this plugin? Only load plugins from people you trust.", "message.plugin_reload": "Reloaded %0 local plugins", + "message.install_plugin": "Installing the plugin %0", "message.preset_no_info": "Preset does not contain information for this slot", "message.restart_to_update": "Restart Blockbench to apply changes", "message.save_file": "Saved as %0", @@ -154,6 +158,9 @@ "message.rename_animation": "Rename Animation", "message.animation_update_var": "Animation Update Variable", "message.untextured": "This surface does not have a texture", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", "message.no_animation_selected": "You have to select an animation to do this", "message.bone_material": "Change bone material", @@ -161,6 +168,9 @@ "message.duplicate_groups.title": "Bone Name Duplicate", "message.duplicate_groups.message": "The name of this bone exists on multiple bones. This can cause problems.", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.project.title": "Project", "dialog.project.name": "File Name", "dialog.project.parent": "Parent Model", @@ -233,12 +243,14 @@ "dialog.create_texture.folder": "Folder", "dialog.create_texture.template": "Template", "dialog.create_texture.compress": "Compress Template", + "dialog.create_texture.power": "Power-of-2 Size", "dialog.create_texture.resolution": "Resolution", "dialog.create_gif.title": "Record GIF", "dialog.create_gif.length": "Length (Seconds)", "dialog.create_gif.fps": "FPS", "dialog.create_gif.compression": "Compression Amount", + "dialog.create_gif.turn": "Turntable Speed", "dialog.create_gif.play": "Start Animation", "dialog.shift_uv.title": "Shift UV", @@ -258,6 +270,13 @@ "dialog.update.installed": "Installed Version", "dialog.update.update": "Update", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "dialog.settings.settings": "Settings", "dialog.settings.keybinds": "Keybindings", "dialog.settings.layout": "Layout", @@ -340,6 +359,8 @@ "settings.transparency.desc": "Render transparent textures transparent", "settings.texture_fps": "Animated Texture FPS", "settings.texture_fps.desc": "Frames per second for animated textures", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", "settings.base_grid": "Small Grid", "settings.base_grid.desc": "Show small grid and axes", @@ -394,6 +415,8 @@ "settings.obj_textures.desc": "Export textures when exporting OBJ file", "settings.credit": "Credit Comment", "settings.credit.desc": "Add a credit comment to exported files", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", "settings.default_path": "Default Path", "settings.default_path.desc": "Folder from where Blockbench loads default textures", "settings.image_editor": "Image Editor", @@ -455,8 +478,6 @@ "action.fill_mode.face": "Face", "action.fill_mode.color": "Color", "action.fill_mode.cube": "Cube", - "action.brush_color": "Color", - "action.brush_color.desc": "Color of the brush", "action.slider_brush_size": "Size", "action.slider_brush_size.desc": "Radius of the brush in pixels", "action.slider_brush_opacity": "Opacity", @@ -533,7 +554,9 @@ "action.export_optifine_full": "Export OptiFine JEM", "action.export_optifine_full.desc": "Export a full OptiFine entity model", "action.export_obj": "Export OBJ Model", - "action.export_obj.desc": "Export a Wavefront OBJ model for use in other programs or to upload to Sketchfab", + "action.export_obj.desc": "Export a Wavefront OBJ model for rendering or game engines", + "action.upload_sketchfab": "Sketchfab Upload", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", "action.save": "Save", "action.save.desc": "Save the current model and textures", "action.settings_window": "Settings...", @@ -546,6 +569,8 @@ "action.donate.desc": "Donate to Blockbench", "action.action_control": "Action Control", "action.action_control.desc": "Search and execute any available action", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", "action.reset_keybindings": "Reset Keybindings", "action.reset_keybindings.desc": "Reset all keybindings to Blockbench's defaults", @@ -591,6 +616,8 @@ "action.sort_outliner.desc": "Sort the outliner alphabetically", "action.local_move": "Move Relative", "action.local_move.desc": "Move rotated elements on their own axes if possible", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", "action.select_window": "Select...", "action.select_window.desc": "Search and select cubes based on their properties", "action.invert_selection": "Invert Selection", @@ -749,6 +776,8 @@ "action.slider_animation_length.desc": "Change the length of the selected animation", "action.slider_keyframe_time": "Timecode", "action.slider_keyframe_time.desc": "Change the timecode of the selected keyframes", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", "action.select_all_keyframes": "Select All Keyframes", "action.select_all_keyframes.desc": "Select all keyframes of the current bone", "action.delete_keyframes": "Delete Keyframes", @@ -760,6 +789,7 @@ "action.next_keyframe": "Next Keyframe", "action.next_keyframe.desc": "Jump to the next keyframe", + "timeline.rotation": "Rotation", "timeline.position": "Position", "timeline.scale": "Scale", @@ -859,9 +889,15 @@ "switches.mirror": "Mirror UV", "switches.autouv": "Auto UV", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "panel.uv": "UV", "panel.display": "Display", "panel.textures": "Textures", + "panel.color": "Color", "panel.outliner": "Outliner", "panel.animations": "Animations", "panel.keyframe": "Keyframe", @@ -906,6 +942,20 @@ "direction.top": "Top", "direction.bottom": "Bottom", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session", + "display.slot.third_right": "Thirdperson Right", "display.slot.third_left": "Thirdperson Left", "display.slot.first_right": "Firstperson Right", diff --git a/lang/es.json b/lang/es.json index 0ff9d077..ba2f379a 100644 --- a/lang/es.json +++ b/lang/es.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Mover origen en el eje Z", "action.brush_mode": "Modo Pincel", "action.brush_mode.desc": "Modo del pincel", - "action.brush_color": "Color", - "action.brush_color.desc": "Color del pincel", "action.slider_brush_size": "Tamaño", "action.slider_brush_size.desc": "Radio del pincel en píxeles", "action.slider_brush_opacity": "Opacidad", @@ -377,7 +375,7 @@ "action.export_optifine_full": "Exportar a OptiFine JEM", "action.export_optifine_full.desc": "Exportar un modelo completo de entidad de OptiFine", "action.export_obj": "Exportar Modelo OBJ", - "action.export_obj.desc": "Crear un modelo WaveFront OBJ para usar en otros programas o para subir a Sketchfab", + "action.export_obj.desc": "Crear un modelo WaveFront OBJ para usar en otros programas", "action.save": "Guardar", "action.save.desc": "Guardar el modelo actual y las texturas", "action.settings_window": "Ajustes...", @@ -586,7 +584,6 @@ "panel.outliner": "Esquema", "panel.options": "Rotación", "panel.options.angle": "Ángulo", - "panel.options.origin": "Origen", "uv_editor.title": "Editor de UV", "uv_editor.all_faces": "Todo", "uv_editor.no_faces": "Ninguno", @@ -849,10 +846,58 @@ "action.previous_keyframe.desc": "Salta al frame anterior", "action.next_keyframe": "Frame Siguiente", "action.next_keyframe.desc": "Salta al frame siguiente", - "message.outdated_client.title": "Outdated client", - "message.outdated_client.message": "Please update to the latest version of Blockbench to do this.", - "action.export_bbmodel": "Export Blockbench Project", - "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", - "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "message.outdated_client.title": "Programa desactualizado", + "message.outdated_client.message": "Por favor actualiza a la última versión de Blockbench para hacer esto.", + "action.export_bbmodel": "Exportar Proyecto de Blockbench", + "action.export_bbmodel.desc": "Exporta un proyecto de Blockbench con todos los cubos, texturas y animaciones", + "action.export_asset_archive": "Descargar Archivo", + "action.export_asset_archive.desc": "Descarga un archivo con el modelo y todas las texturas en él", + "action.upload_sketchfab": "Subir a Sketchfab", + "message.sketchfab.name_or_token": "Por favor, introduce tu token de Sketchfab y un nombre", + "dialog.sketchfab_uploader.title": "Subir Modelo de Sketchfab", + "dialog.sketchfab_uploader.token": "Token de API", + "dialog.sketchfab_uploader.about_token": "El token es usado para conectar Blockbench a tu cuenta de Sketchfab. Puedes encontrarlo en sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Nombre del Modelo", + "dialog.sketchfab_uploader.description": "Descripción", + "dialog.sketchfab_uploader.tags": "Etiquetas", + "settings.sketchfab_token": "Token de Sketchfab", + "settings.sketchfab_token.desc": "Token para autorizar a Blockbench a subir a tu cuenta de Sketchfab", + "panel.color": "Color", + "data.origin": "Origen", + "message.sketchfab.success": "Modelo subido con éxito", + "message.sketchfab.error": "La subida del modelo a Sketchfab ha fallado", + "settings.outliner_colors": "Colores del Borde", + "settings.outliner_colors.desc": "Muestra los colores de cubo en el borde", + "action.upload_sketchfab.desc": "Subir tu modelo a Sketchfab", + "action.element_colors": "Colores de Cubo", + "action.element_colors.desc": "Muestra los colores de cubo en el borde", + "texture.error.file": "Archivo no encontrado", + "texture.error.invalid": "Archivo inválido", + "texture.error.ratio": "Aspecto de ratio inválido", + "texture.error.parent": "Archivo de textura proveído por el modelo padre", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/fr.json b/lang/fr.json index 8239bd70..efe3a2ed 100644 --- a/lang/fr.json +++ b/lang/fr.json @@ -198,12 +198,12 @@ "layout.font.headline": "Police de titre", "about.version": "Version:", "about.creator": "Créateur:", - "about.website": "Site internet:", + "about.website": "Traduction française : HookDonn_\nSite internet:", "about.bugtracker": "Logiciel de suivi des bugs", "about.electron": "Cette application est construite avec Electron, un cadre pour la création d'applications natives avec des technologies Web telles que Javascript, HTML et CSS.", "about.vertex_snap": "Vertex Snapping est basé sur un plugin de SirBenet", "about.icons": "Packs d’icônes :", - "about.libraries": "Bibliothèques", + "about.libraries": "Bibliothèques :", "settings.category.general": "Général", "settings.category.preview": "Prévisualisation ", "settings.category.grid": "Grille", @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Déplacer l’origine sur l’axe Z", "action.brush_mode": "Mode", "action.brush_mode.desc": "Mode peinture", - "action.brush_color": "Couleur", - "action.brush_color.desc": "Couleur du pinceau", "action.slider_brush_size": "Taille", "action.slider_brush_size.desc": "Rayon du pinceau en pixels", "action.slider_brush_opacity": "Opacité", @@ -377,7 +375,7 @@ "action.export_optifine_full": "Exporter OptiFine JEM", "action.export_optifine_full.desc": "Exporter un modèle complet d'entité OptiFine", "action.export_obj": "Exporter un modèle OBJ", - "action.export_obj.desc": "Exporter un modèle OBJ Wavefront pour l'utiliser dans d'autres programmes ou pour le télécharger dans Sketchfab", + "action.export_obj.desc": "Exporter un modèle OBJ Wavefront pour l'utiliser dans d'autres programmes", "action.save": "Sauvegarder", "action.save.desc": "Enregistrer le modèle actuel et les textures", "action.settings_window": "Réglages...", @@ -586,7 +584,6 @@ "panel.outliner": "Liste des blocs", "panel.options": "Rotation", "panel.options.angle": "Angle", - "panel.options.origin": "Origne", "uv_editor.title": "Éditeur UV", "uv_editor.all_faces": "Toutes", "uv_editor.no_faces": "Aucun", @@ -819,40 +816,88 @@ "action.color_picker.desc": "Outil pour choisir la couleur des pixels sur votre texture", "action.open_backup_folder": "Ouvrir le dossier de sauvegarde", "action.open_backup_folder.desc": "Ouvre le dossier de sauvegarde de Blockbench", - "switches.mirror": "Mirror UV", - "language_name": "English", - "message.plugin_reload": "Reloaded %0 local plugins", - "settings.brightness": "Brightness", - "settings.brightness.desc": "Brightness of the preview. Default is 50", - "menu.preview.perspective.reset": "Reset Camera", - "action.fill_mode": "Fill Mode", - "action.fill_mode.desc": "Mode of the fill tool", + "switches.mirror": "Miroir UV", + "language_name": "Anglais", + "message.plugin_reload": "Recharger %0 plugins locaux", + "settings.brightness": "Luminosité", + "settings.brightness.desc": "Luminosité de l'aperçu. La valeur par défaut est 50", + "menu.preview.perspective.reset": "Réinitialiser la camera", + "action.fill_mode": "Mode de remplissage", + "action.fill_mode.desc": "Mode de l'outil de remplissage", "action.fill_mode.face": "Face", - "action.fill_mode.color": "Color", + "action.fill_mode.color": "Couleur", "action.fill_mode.cube": "Cube", - "action.toggle_mirror_uv": "Mirror UV", - "action.toggle_mirror_uv.desc": "Toggle the UV mirroring on the X axis of the selected cubes.", - "action.toggle_uv_overlay": "Toggle UV Overlay", - "action.toggle_uv_overlay.desc": "When enabled, displays all UV mapping overlays above the texture.", - "menu.texture.blank": "Apply to Untextured Faces", - "dialog.scale.select_overflow": "Select Overflow", - "dialog.create_texture.compress": "Compress Template", - "action.action_control": "Action Control", - "action.action_control.desc": "Search and execute any available action", - "keybindings.recording": "Recording Keybinding", - "keybindings.press": "Press a key or key combination or click anywhere on the screen to record your keybinding.", - "action.pivot_tool": "Pivot Tool", - "action.pivot_tool.desc": "Tool to change the pivot point of cubes and bones", - "action.slider_animation_speed": "Playback Speed", - "action.slider_animation_speed.desc": "Playback speed of the timeline in percent", - "action.previous_keyframe": "Previous Keyframe", - "action.previous_keyframe.desc": "Jump to the previous keyframe", - "action.next_keyframe": "Next Keyframe", - "action.next_keyframe.desc": "Jump to the next keyframe", - "message.outdated_client.title": "Outdated client", - "message.outdated_client.message": "Please update to the latest version of Blockbench to do this.", - "action.export_bbmodel": "Export Blockbench Project", - "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", - "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.toggle_mirror_uv": "Miroir UV", + "action.toggle_mirror_uv.desc": "Changer la mise en miroir UV sur l'axe X des cubes sélectionnés.", + "action.toggle_uv_overlay": "Changer le recouvrement UV", + "action.toggle_uv_overlay.desc": "Quand c'est activé, affiche toutes les superpositions de cartographie UV au-dessus de la texture.", + "menu.texture.blank": "Appliquer aux faces non texturées", + "dialog.scale.select_overflow": "Sélectionnez débordement", + "dialog.create_texture.compress": "Compresser le modèle", + "action.action_control": "Contrôle d'action", + "action.action_control.desc": "Rechercher et exécuter toute action disponible", + "keybindings.recording": "Enregistrement des touches de clavier", + "keybindings.press": "Appuyez sur une touche ou une combinaison de touches ou cliquez n'importe où sur l'écran pour enregistrer votre raccourci clavier.", + "action.pivot_tool": "Outil de pivot", + "action.pivot_tool.desc": "Outil pour changer le point de pivot des cubes et des os", + "action.slider_animation_speed": "Vitesse de lecture", + "action.slider_animation_speed.desc": "Vitesse de lecture de la chronologie en pourcentage", + "action.previous_keyframe": "Image clé précédente", + "action.previous_keyframe.desc": "Aller à l'image clé précédente", + "action.next_keyframe": "Image clé suivante", + "action.next_keyframe.desc": "Passer à l'image clé suivante", + "message.outdated_client.title": "Client plus à jour", + "message.outdated_client.message": "Veuillez effectuer la mise à jour vers la dernière version de Blockbench.", + "action.export_bbmodel": "Projet d'exportation Blockbench", + "action.export_bbmodel.desc": "Exporter un projet Blockbench avec tous les cubes, textures et animations", + "action.export_asset_archive": "Télécharger une archive", + "action.export_asset_archive.desc": "Télécharger une archive avec le modèle et toutes les textures qu'elle contient", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Couleur\n", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/ja.json b/lang/ja.json index cd5e2fb1..9f6c0420 100644 --- a/lang/ja.json +++ b/lang/ja.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "原点をZ軸上に移動する", "action.brush_mode": "ブラシモード", "action.brush_mode.desc": "ブラシモード", - "action.brush_color": "カラー", - "action.brush_color.desc": "ブラシのカラー", "action.slider_brush_size": "サイズ", "action.slider_brush_size.desc": "ブラシの半径", "action.slider_brush_opacity": "不透明度", @@ -586,7 +584,6 @@ "panel.outliner": "Outliner", "panel.options": "回転", "panel.options.angle": "角度", - "panel.options.origin": "原点", "uv_editor.title": "UVエディタ", "uv_editor.all_faces": "すべて", "uv_editor.no_faces": "なし", @@ -854,5 +851,53 @@ "action.export_bbmodel": "Export Blockbench Project", "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.export_asset_archive.desc": "Download an archive with the model and all textures in it", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Color", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/nl.json b/lang/nl.json index a2e03686..4523d409 100644 --- a/lang/nl.json +++ b/lang/nl.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Beweeg Oorsprong op de Z-as", "action.brush_mode": "Mode", "action.brush_mode.desc": "Modus van de kwast", - "action.brush_color": "Kleur", - "action.brush_color.desc": "Kleur van de kwast", "action.slider_brush_size": "Grootte", "action.slider_brush_size.desc": "Formaat van de kwast", "action.slider_brush_opacity": "Doorzichtigheid", @@ -377,7 +375,7 @@ "action.export_optifine_full": "Exporteer OptiFine JEM", "action.export_optifine_full.desc": "Exporteer een volledig OptiFine entity model", "action.export_obj": "Exporteer OBJ Model", - "action.export_obj.desc": "Exporteer een Wavefront OBJ model om in andere programma's te gebuiken of om te uploaden naar Sketchfab", + "action.export_obj.desc": "Exporteer een Wavefront OBJ model om in andere programma's te gebuiken", "action.save": "Opslaan", "action.save.desc": "Het huidige model en texturen opslaan", "action.settings_window": "Instellingen...", @@ -586,7 +584,6 @@ "panel.outliner": "Outliner", "panel.options": "Rotatie", "panel.options.angle": "Hoek", - "panel.options.origin": "Oorsprong", "uv_editor.title": "UV Editor", "uv_editor.all_faces": "Alle", "uv_editor.no_faces": "Geen", @@ -728,15 +725,15 @@ "action.move_down": "Beweeg naar beneden", "action.move_down.desc": "Beweeg de geselecteerde kubussen naar beneden ten opzichte van de huidige camerahoek", "action.move_left": "Beweeg Naar Links", - "action.move_left.desc": "Move the selected cubes left relative to the current camera angle", + "action.move_left.desc": "Beweeg de geselecteerde kubussen naar links ten opzichte van de huidige camera hoek", "action.move_right": "Beweeg Naar Rechts", - "action.move_right.desc": "Move the selected cubes right relative to the current camera angle", + "action.move_right.desc": "Beweeg de geselecteerde kubussen naar rechts ten opzichte van de huidige camera hoek", "action.move_forth": "Beweeg naar voren", - "action.move_forth.desc": "Move the selected cubes forth relative to the current camera angle", + "action.move_forth.desc": "Beweeg de geselecteerde kubussen naar voren ten opzichte van de huidige camera hoek", "action.move_back": "Beweeg naar achteren", - "action.move_back.desc": "Move the selected cubes back relative to the current camera angle", + "action.move_back.desc": "Beweeg de geselecteerde kubussen naar achteren ten opzichte van de huidige camera hoek", "layout.color.wireframe": "Wireframe", - "layout.color.wireframe.desc": "Wireframe view lines", + "layout.color.wireframe.desc": "Wireframe bekijk lijnen", "action.add_animation": "Voeg Animatie Toe", "action.add_animation.desc": "Creëer een blanco animatie ", "action.load_animation_file": "Import Animatie", @@ -760,7 +757,7 @@ "message.rename_animation": "Hernoem Animatie", "message.animation_update_var": "Animatie Update Variabele", "message.no_animation_selected": "Je moet een animatie selecteren om dit te doen", - "message.no_bone_selected": "You have to select a bone to do this", + "message.no_bone_selected": "Je moet een bot selecteren om dit te doen", "message.duplicate_groups.title": "Bone Name Duplicate", "message.duplicate_groups.message": "The name of this bone exists on multiple bones. This can cause problems.", "action.select_all_keyframes": "Select All Keyframes", @@ -854,5 +851,53 @@ "action.export_bbmodel": "Export Blockbench Project", "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.export_asset_archive.desc": "Download an archive with the model and all textures in it", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Color", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/pl.json b/lang/pl.json index a95320a7..be62063c 100644 --- a/lang/pl.json +++ b/lang/pl.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Przesuń punkt obrotu na osi Z", "action.brush_mode": "Tryb pędzla", "action.brush_mode.desc": "Tryb pędzla", - "action.brush_color": "Kolor", - "action.brush_color.desc": "Kolor pędzla", "action.slider_brush_size": "Wielkość", "action.slider_brush_size.desc": "Promień pędzla w pikselach", "action.slider_brush_opacity": "Nieprzezroczystość", @@ -586,7 +584,6 @@ "panel.outliner": "Outliner", "panel.options": "Rotacja", "panel.options.angle": "Kąt", - "panel.options.origin": "Punkt obrotu", "uv_editor.title": "Edytor UV", "uv_editor.all_faces": "Wszystkie", "uv_editor.no_faces": "Żadne", @@ -853,6 +850,54 @@ "message.outdated_client.message": "Please update to the latest version of Blockbench to do this.", "action.export_bbmodel": "Export Blockbench Project", "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", - "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.export_asset_archive": "Pobier Archiv", + "action.export_asset_archive.desc": "Download an archive with the model and all textures in it", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Color", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/pt.json b/lang/pt.json new file mode 100644 index 00000000..df22b9c0 --- /dev/null +++ b/lang/pt.json @@ -0,0 +1,903 @@ +{ + "dialog.ok": "Ok", + "dialog.cancel": "Cancelar", + "dialog.confirm": "Confirmar", + "dialog.close": "Fechar", + "dialog.import": "Importar", + "dialog.save": "Salvar", + "dialog.discard": "Descartar", + "dialog.dontshowagain": "Não mostrar novamente", + "data.cube": "Cubo", + "data.cubes": "Cubos", + "data.group": "Grupo", + "data.texture": "Textura", + "data.plugin": "Plugin", + "data.preview": "Pré-visualização", + "data.toolbar": "Barra de ferramentas", + "data.image": "Imagem", + "keys.ctrl": "Control", + "keys.shift": "Shift", + "keys.alt": "Alt", + "keys.meta": "Cmd", + "keys.delete": "Delete", + "keys.space": "Espaço", + "keys.leftclick": "Clique esquerdo", + "keys.middleclick": "Clique do meio", + "keys.rightclick": "Clique direito", + "keys.tab": "Tab", + "keys.backspace": "Backspace", + "keys.enter": "Enter", + "keys.escape": "Esc", + "keys.function": "F%0", + "keys.numpad": "Numpad %0", + "keys.caps": "Capslock", + "keys.menu": "Menu de contexto", + "keys.left": "Esquerda", + "keys.up": "Acima", + "keys.right": "Direita", + "keys.down": "Abaixo", + "keys.pageup": "Page Up", + "keys.pagedown": "Page Down", + "keys.plus": "Mais", + "keys.comma": "Vírgula", + "keys.point": "Ponto", + "keys.minus": "Menos", + "keys.cross": "Cruz", + "keys.end": "End", + "keys.pos1": "Pos 1", + "keys.printscreen": "Print Screen", + "keys.pause": "Pausa", + "message.rotation_limit.title": "Limites de rotação", + "message.rotation_limit.message": "As rotações são limitadas pelo Minecraft para um eixo e incrementos de 22.5 graus. Girar em um eixo diferente irá limpar todas as rotações nos outros eixos. Desative a opção \"Rotação Restrita\" se estiver modelando para outros propósitos e precisar de rotações livres.", + "message.file_not_found.title": "Arquivo não encontrado", + "message.file_not_found.message": "Blockbench não pode encontrar o arquivo solicitado. Verifique se ele está salvo localmente e não em uma núvem.", + "message.screenshot.title": "Captura de tela", + "message.screenshot.message": "Captura de tela capturada.", + "message.screenshot.clipboard": "Prancheta", + "message.screenshot.right_click": "Captura de tela - Botão direito do mouse para copiar", + "message.invalid_file.title": "Arquivo inválido", + "message.invalid_file.message": "Não foi possível abrir o arquivo json: %0", + "message.invalid_model.title": "Arquivo de modelo inválido", + "message.invalid_model.message": "Este arquivo não contém dados de modelo válidos.", + "message.child_model_only.title": "Modelo Filho Vazio\n", + "message.child_model_only.message": "Este arquivo é filho de %0 e não contém nenhum modelo", + "message.drag_background.title": "Posicionar fundo.", + "message.drag_background.message": "Arraste o fundo para mover sua posição. Segure a tecla shift e arraste para cima e para baixo para alterar seu tamanho.", + "message.unsaved_textures.title": "Texturas não salvas", + "message.unsaved_textures.message": "Seu modelo tem texturas não salvas. Certifique-se de salvá-las e colá-las em seu pacote de recursos na pasta correta.", + "message.model_clipping.title": "Modelo muito grande", + "message.model_clipping.message": "Seu modelo contém %0 cubos que são maiores do que o limite 3x3x3 permitido pelo Minecraft. Esse modelo não irá funcionar no Minecraft. Habilite a opção 'Limites restrita' para prevenir isso.", + "message.loose_texture.title": "Importar textura", + "message.loose_texture.message": "A textura importada não faz parte de um pacote de recursos. Minecraft apenas consegue carregar texturas dentro da pasta de texturas de um pacote de recursos carregado.", + "message.loose_texture.change": "Mudar caminho", + "message.update_res.title": "Resolução da textura", + "message.update_res.message": "Você gostaria de atualizar a resolução do projeto para a resolução dessa textura? Clique em 'Cancelar' se sua textura tem uma resolução maior que o normal.", + "message.update_res.update": "Atualizar", + "message.bedrock_overwrite_error.message": "Blockbench não pode combinar esse modelo com o arquivo antigo", + "message.bedrock_overwrite_error.backup_overwrite": "Criar backup e sobrescrever", + "message.bedrock_overwrite_error.overwrite": "Sobrescrever", + "message.close_warning.message": "Você deseja salvar o seu modelo?", + "message.close_warning.web": "O seu trabalho atual será perdido. Tem certeza que deseja sair?", + "message.default_textures.title": "Texturas padrão", + "message.default_textures.message": "Selecione a pasta \"textures\" do pacote de recursos padrão", + "message.default_textures.detail": "Extraia o pacote de texturas padrão do jar do Minecraft ou do Google e baixe-o. Em seguida, localize a pasta \"texturas\" e abra-a. O Blockbench lembrará esse local e tentará buscar texturas a partir dele se não puder encontrá-las no pacote de texturas atual.", + "message.default_textures.select": "Selecione a pasta das \"texturas\" padrão", + "message.default_textures.continue": "Continuar", + "message.default_textures.remove": "Remover", + "message.image_editor.title": "Selecione um editor de imagens", + "message.image_editor.file": "Selecionar Arquivo...", + "message.image_editor.exe": "Selecione o executável do editor de imagens", + "message.display_skin.title": "Exibir Skin", + "message.display_skin.message": "Selecione um arquivo de skin do seu computador ou digite um nome de jogador", + "message.display_skin.upload": "Carregar Skin", + "message.display_skin.name": "Nome de usuário", + "message.display_skin.reset": "Resetar", + "message.invalid_plugin": "Plugin inválido, veja o console", + "message.load_plugin_app": "Você quer permitir esse plugin de fazer alterações em seu computador? Apenas carregue plugins de pessoas que você confia.", + "message.load_plugin_web": "Você quer carregar este plugin? Apenas carregue plugins de pessoas que você confia.", + "message.preset_no_info": "A predefinição não contém informações para este espaço", + "message.restart_to_update": "Reinicie o Blockbench para aplicar as mudanças", + "message.save_file": "Salvo como %0", + "message.save_obj": "Salvo como modelo .obj", + "message.save_entity": "Salvo como modelo de entidade do \"bedrock\"", + "message.rename_cubes": "Renomear Cubos", + "dialog.project.title": "Projeto", + "dialog.project.name": "Nome do Arquivo", + "dialog.project.parent": "Modelo Pai", + "dialog.project.geoname": "Nome da Geometria do Mob", + "dialog.project.openparent": "Abrir Pai", + "dialog.project.ao": "Oclusão de ambiente", + "dialog.project.texture_size": "Tamanho da Textura", + "dialog.project.width": "Largura", + "dialog.project.height": "Altura", + "dialog.project.to_blockmodel": "Para Modelo de Bloco", + "dialog.project.to_entitymodel": "Para Modelo de Entidade", + "dialog.texture.title": "Textura", + "dialog.texture.name": "Nome", + "dialog.texture.variable": "Variável", + "dialog.texture.namespace": "Namespace", + "dialog.texture.folder": "Pasta", + "dialog.extrude.title": "Extrudar (expulsar) Imagem", + "dialog.extrude.mode": "Modo de Digitalização", + "dialog.extrude.mode.areas": "Areas", + "dialog.extrude.mode.lines": "Linhas", + "dialog.extrude.mode.columns": "Colunas", + "dialog.extrude.mode.pixels": "Pixels", + "dialog.extrude.opacity": "Opacidade Mínima", + "dialog.extrude.scan": "Digitalizar e Importar", + "dialog.display_preset.title": "Criar Predefinição", + "dialog.display_preset.message": "Selecione os slots que você deseja salvar", + "dialog.display_preset.create": "Criar", + "dialog.select.title": "Selecione", + "dialog.select.new": "Nova Seleção", + "dialog.select.group": "No Grupo Selecionado", + "dialog.select.name": "Nome Contém", + "dialog.select.random": "Aleatório", + "dialog.select.select": "Selecione", + "dialog.scale.title": "Modelo Escala", + "dialog.scale.axis": "Eixo", + "dialog.scale.scale": "Escala", + "dialog.scale.clipping": "Corte de modelo: seu modelo é muito grande para a tela", + "dialog.scale.confirm": "Escala", + "dialog.plugins.title": "Plug-ins", + "dialog.plugins.installed": "Instalado", + "dialog.plugins.available": "Disponível", + "dialog.plugins.install": "Instalar", + "dialog.plugins.uninstall": "Desinstalar", + "dialog.plugins.reload": "Recarregar", + "dialog.plugins.none_installed": "Nenhum plug-in instalado", + "dialog.plugins.none_available": "Nenhum plug-in disponível", + "dialog.plugins.outdated": "Requer uma versão mais nova do Blockbench", + "dialog.plugins.web_only": "Apenas para o aplicativo da web", + "dialog.plugins.app_only": "Apenas para o aplicativo do desktop", + "dialog.plugins.author": "por %0", + "dialog.plugins.show_less": "Mostrar Mais", + "dialog.entitylist.title": "Abrir Modelo de Entidade", + "dialog.entitylist.text": "Selecione o modelo que você deseja importar", + "dialog.entitylist.bones": "Ossos", + "dialog.entitylist.cubes": "Cubos", + "dialog.create_texture.title": "Criar Textura", + "dialog.create_texture.name": "Nome", + "dialog.create_texture.folder": "Pasta", + "dialog.create_texture.template": "Modelo", + "dialog.create_texture.resolution": "Resolução", + "dialog.input.title": "Entrada", + "dialog.update.title": "Atualizações", + "dialog.update.refresh": "Tente Novamente!", + "dialog.update.up_to_date": "Blockbench está atualizado!", + "dialog.update.connecting": "Conectando ao Servidor", + "dialog.settings.settings": "Configurações", + "dialog.settings.keybinds": "Combinações de teclas", + "dialog.settings.layout": "Layout", + "dialog.settings.about": "Sobre", + "layout.color.back": "Voltar", + "layout.color.back.desc": "Planos de fundo e campos de entrada", + "layout.color.dark": "Sombrio", + "layout.color.dark.desc": "Fundo de tela", + "layout.color.ui": "UI", + "layout.color.ui.desc": "Cor da interface principal", + "layout.color.bright_ui": "UI brilhante", + "layout.color.bright_ui.desc": "Menus contextuais e dicas de ferramentas", + "layout.color.button": "Botão", + "layout.color.button.desc": "Botões e interruptores", + "layout.color.selected": "Selecionado", + "layout.color.selected.desc": "Guias e objetos selecionados", + "layout.color.border": "Borda", + "layout.color.border.desc": "Borda de botões e entradas", + "layout.color.accent": "Acento", + "layout.color.accent.desc": "Polegar deslizante e outros detalhes", + "layout.color.grid": "Grade", + "layout.color.grid.desc": "Grade de visualização 3D", + "layout.color.text": "Texto", + "layout.color.text.desc": "Texto Normal", + "layout.color.light": "Luz", + "layout.color.light.desc": "Texto destacado", + "layout.color.accent_text": "Texto Acentuado", + "layout.color.accent_text.desc": "Texto em elementos brilhantes ou acentuados", + "layout.font.main": "Fonte Principal", + "layout.font.headline": "Fonte de título", + "about.version": "Versão:", + "about.creator": "Criador:", + "about.website": "Website:", + "about.bugtracker": "Rastreador de Problemas:", + "about.electron": "Este aplicativo é construído com o Electron, uma estrutura para criar aplicativos nativos com tecnologias da Web, como Javascript, HTML e CSS.", + "about.vertex_snap": "O Vertex Snapping é baseado em um plugin do SirBenet", + "about.icons": "Pacotes de Ícone", + "about.libraries": "Bibliotecas:", + "settings.category.general": "Geral", + "settings.category.preview": "Pré-Visualização", + "settings.category.grid": "Grade", + "settings.category.edit": "Editar", + "settings.category.snapping": "Snapping", + "settings.category.defaults": "Padrões", + "settings.category.dialogs": "Diálogos", + "settings.category.export": "Exportar", + "settings.language": "Idioma", + "settings.language.desc": "Idioma da Interface. Reinicie o Blockbench para aplicar as alterações", + "settings.show_actions": "Ações de Exibição", + "settings.show_actions.desc": "Exibir todas as ações na barra de status", + "settings.backup_interval": "Intervalo de Backup", + "settings.backup_interval.desc": "Intervalo dos backups automáticos em minutos", + "settings.origin_size": "Origem da Rotação", + "settings.origin_size.desc": "Tamanho da origem da rotação", + "settings.control_size": "Tamanho do controle do eixo", + "settings.control_size.desc": "Tamanho da ferramenta de controle de 3 eixos", + "settings.display_skin": "Exibir Skin", + "settings.display_skin.desc": "Skin usada para o modelo de player de referência de exibição", + "settings.shading": "Sombreamento", + "settings.shading.desc": "Ativar sombreamento", + "settings.transparency": "Transparência", + "settings.transparency.desc": "Renderizar Texturas Transparentes", + "settings.texture_fps": "Textura animada FPS", + "settings.texture_fps.desc": "Quadros por segundo para texturas animadas", + "settings.base_grid": "Grade Pequena", + "settings.base_grid.desc": "Mostrar pequena grade e eixos", + "settings.large_grid": "Grade Grande", + "settings.large_grid.desc": "Mostrar grade de blocos 3x3", + "settings.full_grid": "Grade Grande Completa", + "settings.full_grid.desc": "Mostrar grade precisa de 3x3", + "settings.large_box": "Caixa grande", + "settings.large_box.desc": "Mostrar limites de blocos 3x3", + "settings.display_grid": "Modo de exibição", + "settings.display_grid.desc": "Mostrar grade no modo de exibição", + "settings.undo_limit": "Limite de desfazer", + "settings.undo_limit.desc": "Número de etapas que você pode desfazer", + "settings.restricted_canvas": "Tela restrita", + "settings.restricted_canvas.desc": "Restringir a Tela a uma área de bloco de 3x3 para evitar modelos inválidos", + "settings.limited_rotation": "Rotação restrita", + "settings.limited_rotation.desc": "Restringir rotação aos valores válidos para modelos Minecraft", + "settings.local_move": "Mova-se nos Eixos Relativos", + "settings.local_move.desc": "Mover elementos rotacionados em seus próprios eixos, se possível", + "settings.canvas_unselect": "Clique em Tela Deselecionar", + "settings.canvas_unselect.desc": "Desmarca todos os elementos ao clicar no fundo da tela", + "settings.paint_side_restrict": "Restringir o pincel ao lado", + "settings.paint_side_restrict.desc": "Restringir os pincéis para pintar apenas no lado atual", + "settings.autouv": "Auto UV", + "settings.autouv.desc": "Ativar AutoUV por padrão", + "settings.create_rename": "Renomear Novo Cubo", + "settings.create_rename.desc": "Campo de nome de foco ao criar um novo elemento ou grupo", + "settings.edit_size": "Resolução de Grade", + "settings.edit_size.desc": "Resolução da grade em que os cubos se encaixam", + "settings.shift_size": "Resolução Shift", + "settings.shift_size.desc": "Resolução da grade enquanto mantém Shift pressionado", + "settings.ctrl_size": "Resolução de controle", + "settings.ctrl_size.desc": "Resolução da grade, mantendo o controle", + "settings.negative_size": "Tamanho negativo", + "settings.negative_size.desc": "Permitir que a ferramenta de redimensionamento use tamanhos negativos", + "settings.dialog_unsaved_textures": "Texturas não salvas", + "settings.dialog_unsaved_textures.desc": "Mostrar a caixa de diálogo \"Texturas não salvas\"", + "settings.dialog_larger_cubes": "Modelo muito grande", + "settings.dialog_larger_cubes.desc": "Mostrar a caixa de diálogo \"Modelo muito grande\"", + "settings.dialog_rotation_limit": "Limites de Rotação", + "settings.dialog_rotation_limit.desc": "Mostrar caixa de diálogo \"Limites de rotação\"", + "settings.minifiedout": "Exportação Minificada", + "settings.minifiedout.desc": "Escreva o arquivo JSON em uma linha", + "settings.export_groups": "Grupos de Exportação", + "settings.export_groups.desc": "Salvar grupos em arquivos blockmodel", + "settings.obj_textures": "Texturas de Exportação", + "settings.obj_textures.desc": "Exportar texturas ao exportar arquivo OBJ", + "settings.credit": "Comentário de crédito", + "settings.credit.desc": "Adicionar um comentário de crédito aos arquivos exportados", + "settings.default_path": "Caminho Padrão", + "settings.default_path.desc": "Pasta de onde o Blockbench carrega texturas padrão", + "settings.image_editor": "Editor de Imagem", + "settings.image_editor.desc": "Editor de imagens padrão para editar texturas com", + "category.navigate": "Navegação", + "category.tools": "Ferramentas", + "category.file": "Arquivo", + "category.blockbench": "Blockbench", + "category.edit": "Editar", + "category.transform": "Transformar", + "category.filter": "Filtro", + "category.view": "Visão", + "category.display": "Configurações Padrão", + "category.textures": "Texturas", + "category.misc": "Diversos", + "keybind.preview_select": "Selecione", + "keybind.preview_rotate": "Visão de Rotação", + "keybind.preview_drag": "Visão de Arrastar", + "keybind.confirm": "Confirme", + "keybind.cancel": "Cancelar", + "action.slider_pos_x": "Mover X", + "action.slider_pos_x.desc": "Move cubos no eixo X", + "action.slider_pos_y": "Mover Y", + "action.slider_pos_y.desc": "Move cubos no eixo Y", + "action.slider_pos_z": "Mover Z", + "action.slider_pos_z.desc": "Move cubos no eixo Z", + "action.slider_size_x": "Tamanho X", + "action.slider_size_x.desc": "Redimensionar cubos no eixo X", + "action.slider_size_y": "Tamanho Y", + "action.slider_size_y.desc": "Redimensionar cubos no eixo Y", + "action.slider_size_z": "Tamanho Z", + "action.slider_size_z.desc": "Redimensionar cubos no eixo Z", + "action.slider_inflate": "Aumentar", + "action.slider_inflate.desc": "Aumentar cubos em todas as direções sem alterar o UV.", + "action.slider_rotation_x": "Rodar X", + "action.slider_rotation_x.desc": "Rotaciona cubos no eixo X", + "action.slider_rotation_y": "Rodar Y", + "action.slider_rotation_y.desc": "Rotaciona cubos no eixo X", + "action.slider_rotation_z": "Rodar Z", + "action.slider_rotation_z.desc": "Rotaciona cubos no eixo X", + "action.slider_origin_x": "Origem X", + "action.slider_origin_x.desc": "Move a origem no eixo X", + "action.slider_origin_y": "Origem Y", + "action.slider_origin_y.desc": "Move a origem no eixo X", + "action.slider_origin_z": "Origem Z", + "action.slider_origin_z.desc": "Move a origem no eixo X", + "action.brush_mode": "Modo de Pincel", + "action.brush_mode.desc": "Modo do Pincel", + "action.slider_brush_size": "Tamanho", + "action.slider_brush_size.desc": "Raio do pincel em pixels", + "action.slider_brush_opacity": "Opacidade", + "action.slider_brush_opacity.desc": "Opacidade do pincel em porcentagem", + "action.slider_brush_softness": "Suavidade", + "action.slider_brush_softness.desc": "Suavidade do pincel em porcentagem", + "action.background_color": "Cor de fundo", + "action.background_color.desc": "Cor de fundo da textura criada", + "action.uv_slider_pos_x": "Move Horizontal", + "action.uv_slider_pos_x.desc": "Mova a seleção UV de todos os cubos selecionados horizontalmente", + "action.uv_slider_pos_y": "Move Vertical", + "action.uv_slider_pos_y.desc": "Mova a seleção UV de todos os cubos selecionados verticalmente", + "action.uv_slider_size_x": "Escala Horizontal", + "action.uv_slider_size_x.desc": "Escale a seleção UV de todos os cubos selecionados horizontalmente", + "action.uv_slider_size_y": "Escala Vertical", + "action.uv_slider_size_y.desc": "Escale a seleção UV de todos os cubos selecionados verticalmente", + "action.vertex_snap_mode": "Modo Snap", + "action.vertex_snap_mode.desc": "Selecione se o Vertex Snap mover elementos para a posição selecionada ou redimensioná-los", + "action.move_tool": "Mover", + "action.move_tool.desc": "Ferramenta para selecionar e mover elementos", + "action.resize_tool": "Redimensionar", + "action.resize_tool.desc": "Ferramenta para selecionar e redimensionar elementos", + "action.brush_tool": "Pincel", + "action.brush_tool.desc": "Ferramenta para pintar texturas de bitmap em superfícies ou o editor de UV", + "action.vertex_snap_tool": "Vertex Snap", + "action.vertex_snap_tool.desc": "Mova um cubo para outro cubo conectando dois vértices (origens)", + "action.swap_tools": "Ferramentas de Troca", + "action.swap_tools.desc": "Alternar entre a ferramenta de movimentação e redimensionamento", + "action.project_window": "Projeto...", + "action.project_window.desc": "Abre a janela do projeto, onde você pode editar os metadados do seu modelo", + "action.new_block_model": "Novo Modelo", + "action.new_block_model.desc": "Cria um novo modelo de bloco / item", + "action.new_entity_model": "Novo Modelo de Entidade", + "action.new_entity_model.desc": "Cria um novo modelo de entidade de base", + "action.open_model": "Abrir Modelo", + "action.open_model.desc": "Abre o arquivo do modelo do seu computador.", + "action.add_model": "Adicionar modelo", + "action.add_model.desc": "Adicionar um modelo de um arquivo ao modelo atual", + "action.extrude_texture": "Textura extrudada", + "action.extrude_texture.desc": "Gere um modelo esticando uma textura", + "action.export_blockmodel": "Exportar Blockmodel", + "action.export_blockmodel.desc": "Exportar um bloco ou modelo de item do Minecraft", + "action.export_entity": "Exportar a Entidade Bedrock", + "action.export_entity.desc": "dicione o modelo atual como uma entidade a um arquivo mobs.json", + "action.export_optifine_part": "Exportar OptiFine JPM", + "action.export_optifine_part.desc": "Exportar um modelo de parte da entidade para o OptiFine", + "action.export_optifine_full": "Exportar OptiFine JEM", + "action.export_optifine_full.desc": "Exportar um modelo de entidade completo do OptiFine", + "action.export_obj": "Exportar Modelo OBJ", + "action.export_obj.desc": "Exportar um modelo Wavefront OBJ para uso em outros programas ou fazer upload para o Sketchfab", + "action.save": "Salve", + "action.save.desc": "Salve o modelo e as texturas atuais", + "action.settings_window": "Configurações", + "action.settings_window.desc": "Abre o diálogo de configurações do Blockbench.", + "action.plugins_window": "Plugins...", + "action.plugins_window.desc": "Abre a janela da loja de plugins", + "action.update_window": "Atualizações...", + "action.update_window.desc": "Busca por atualizações do Blockbench", + "action.donate": "Doar", + "action.donate.desc": "Doar para o Blockbench", + "action.reset_keybindings": "Redefinir atalhos de teclado", + "action.reset_keybindings.desc": "Redefinir todas as combinações de teclas com os padrões do Blockbench", + "action.import_layout": "Importar Layout", + "action.import_layout.desc": "Importa um arquivo de layout", + "action.export_layout": "Exportar Layout", + "action.export_layout.desc": "Cria um arquivo de layout baseado nas configurações atuais", + "action.reset_layout": "Redefinir Layout", + "action.reset_layout.desc": "Redefine para layout padrão do Blockbench", + "action.load_plugin": "Carregar Plugin de um Arquivo", + "action.load_plugin.desc": "Carrega um plugin importando o arquivo de origem.", + "action.reload_plugins": "Recarregar Plugins", + "action.reload_plugins.desc": "Recarrega todos os plugins de desenvolvimento.", + "action.uv_dialog": "Janela UV", + "action.uv_dialog.desc": "Abre o diálogo UV para ver todas as faces próximas umas das outras", + "action.uv_dialog_full": "Vista Completa", + "action.uv_dialog_full.desc": "Abra o diálogo UV para editar uma face em tela cheia", + "action.undo": "Desfazer", + "action.undo.desc": "Anula a última alteração", + "action.redo": "Refazer", + "action.redo.desc": "Reverte o último desfazer", + "action.copy": "Copiar", + "action.copy.desc": "Copie a seleção selecionada, face ou configurações de exibição", + "action.paste": "Colar", + "action.paste.desc": "Cole a seleção selecionada, face ou configurações de exibição", + "action.cut": "Cortar", + "action.cut.desc": "Corta a seleção selecionada, face ou configurações de exibição", + "action.add_cube": "Adicionar Cubo", + "action.add_cube.desc": "Adiciona um novo cubo", + "action.add_group": "Adicionar Grupo", + "action.add_group.desc": "Adiciona um novo grupo ou osso", + "action.outliner_toggle": "Alternar Mais Opções", + "action.outliner_toggle.desc": "Alterna opções para mais opções no delineador", + "action.duplicate": "Duplicar", + "action.duplicate.desc": "Duplica os cubos ou grupos selecionados", + "action.delete": "Deletar", + "action.delete.desc": "Exclui os cubos ou grupos selecionados", + "action.sort_outliner": "Ordenar Outliner", + "action.sort_outliner.desc": "Ordenar o delineador em ordem alfabética", + "action.local_move": "Mover Relativo", + "action.local_move.desc": "Mover elementos rotacionados em seus próprios eixos, se possível", + "action.select_window": "Selecionar...", + "action.select_window.desc": "Pesquisa e seleciona cubos com base em suas propriedades", + "action.invert_selection": "Seleção invertida", + "action.invert_selection.desc": "Inverte a seleção atual de cubos", + "action.select_all": "Selecionar todos", + "action.select_all.desc": "Seleciona todos os cubos", + "action.collapse_groups": "Recolher grupos", + "action.collapse_groups.desc": "Recolhe todos os grupos", + "action.scale": "Escala...", + "action.scale.desc": "Escala os cubos selecionados", + "action.rotate_x_cw": "Rodar CW", + "action.rotate_x_cw.desc": "Rodar os cubos selecionados a 90 ° no eixo X", + "action.rotate_x_ccw": "Rodar Counter-CW", + "action.rotate_x_ccw.desc": "Gira os cubos selecionados -90 ° no eixo X", + "action.rotate_y_cw": "Rodar CW", + "action.rotate_y_cw.desc": "Rodar os cubos selecionados a 90 ° no eixo Y", + "action.rotate_y_ccw": "Rodar Counter-CW", + "action.rotate_y_ccw.desc": "Gira os cubos selecionados -90 ° no eixo Y", + "action.rotate_z_cw": "Rodar CW", + "action.rotate_z_cw.desc": "Roda os cubos selecionados a 90 ° no eixo Z", + "action.rotate_z_ccw": "Rodar Counter-CW", + "action.rotate_z_ccw.desc": "Roda os cubos selecionados -90 ° no eixo Z", + "action.flip_x": "Girar X", + "action.flip_x.desc": "Gira os cubos selecionados no eixo X", + "action.flip_y": "Girar Y", + "action.flip_y.desc": "Gira os cubos selecionados no eixo Y", + "action.flip_z": "Girar Z", + "action.flip_z.desc": "Gira os cubos selecionados no eixo Z", + "action.center_x": "Centralizar X", + "action.center_x.desc": "Centraliza os cubos selecionados no eixo X", + "action.center_y": "Centralizar Y", + "action.center_y.desc": "Centraliza os cubos selecionados no eixo Y", + "action.center_z": "Centralizar Z", + "action.center_z.desc": "Centraliza os cubos selecionados no eixo Z", + "action.center_all": "Centralizar Todos", + "action.center_all.desc": "Centraliza os cubos selecionados", + "action.toggle_visibility": "Alternar visibilidade", + "action.toggle_visibility.desc": "Alterna a visibilidade dos cubos selecionados.", + "action.toggle_export": "Alternar exportação", + "action.toggle_export.desc": "Alterna as configurações de exportação dos cubos selecionados.", + "action.toggle_autouv": "Alternar Auto UV", + "action.toggle_autouv.desc": "Alternas as configurações de auto UV dos cubos selecionados.", + "action.toggle_shade": "Alternar sombreamento", + "action.toggle_shade.desc": "Alterna o sombreamento dos cubos selecionados.", + "action.rename": "Renomear", + "action.rename.desc": "Muda o nome dos cubos selecionados.", + "action.add_display_preset": "Nova predefinição", + "action.add_display_preset.desc": "Adicionar uma nova predefinição de configuração de exibição.", + "action.fullscreen": "Tela Cheia", + "action.fullscreen.desc": "Alterna para o modo de Tela Cheia.", + "action.zoom_in": "Mais Zoom", + "action.zoom_in.desc": "Aumentar o zoom para ampliar a interface.", + "action.zoom_out": "Reduzir o Zoom", + "action.zoom_out.desc": "Reduzir o Zoom para reduzir a interface.", + "action.zoom_reset": "Resetar Zoom", + "action.zoom_reset.desc": "Reseta o Zoom para padrão 100%.", + "action.reset_interface": "Resetar Interface", + "action.reset_interface.desc": "Reseta o tamanho e as posições da Interface (GUI)", + "action.toggle_wireframe": "Alterar Wireframe", + "action.toggle_wireframe.desc": "Altera o wireframe para o modo padrão.", + "action.screenshot_model": "Capturar Modelo", + "action.screenshot_model.desc": "Tira uma captura de tela recortada do modelo do ângulo atual", + "action.screenshot_app": "Capturar Aplicativo", + "action.screenshot_app.desc": "Tira uma captura de tela de toda a aplicação", + "action.toggle_quad_view": "Alternar vista quad", + "action.toggle_quad_view.desc": "Alternar o modo de 4 pontos de vista", + "action.import_texture": "Importar Textura", + "action.import_texture.desc": "Importa uma ou mais texturas do seu arquivo do sistema", + "action.create_texture": "Criar Textura", + "action.create_texture.desc": "Cria uma textura ou modelo de textura em branco", + "action.reload_textures": "Recarregar Texturas", + "action.reload_textures.desc": "Recarrega todas as texturas", + "action.save_textures": "Salvar Texturas", + "action.save_textures.desc": "Salva as texturas não salvas", + "action.animated_textures": "Iniciar Texturas Animadas", + "action.animated_textures.desc": "Inicia e Pausa a pré visualização das texturas animadas", + "action.origin_to_geometry": "Origem para Geometria", + "action.origin_to_geometry.desc": "Definir a origem para o centro da geometria", + "action.rescale_toggle": "Escala de alternância", + "action.rescale_toggle.desc": "Rescalar cubos com base em sua rotação atual", + "action.bone_reset_toggle": "Resetar Osso", + "action.bone_reset_toggle.desc": "Impedir que o osso exiba cubos do modelo pai", + "action.reload": "Recarregar o Blockbench", + "action.reload.desc": "Recarrega o Blockbench. Isso removerá todos os progressos não salvos", + "menu.file": "Arquivo", + "menu.edit": "Editar", + "menu.transform": "Transformar", + "menu.filter": "Filtro", + "menu.display": "Padrão", + "menu.view": "Visão", + "menu.file.new": "Novo", + "menu.file.recent": "Recente", + "menu.file.import": "Importar", + "menu.file.export": "Exportar", + "menu.transform.rotate": "Rodar", + "menu.transform.flip": "Girar", + "menu.transform.center": "Centralizar", + "menu.transform.properties": "Propiedades", + "menu.display.preset": "Aplicar predefinição", + "menu.display.preset_all": "Aplica Predefinição em toda a parte", + "menu.display.remove_preset": "Remover Predefinição", + "menu.view.zoom": "Zoom", + "menu.view.background": "Fundo", + "menu.view.screenshot": "Captura de Tela", + "menu.cube.duplicate": "Duplicar", + "menu.cube.color": "Criador de Cor", + "menu.cube.texture": "Textura", + "menu.cube.texture.transparent": "Transparente", + "menu.cube.texture.blank": "Em branco", + "menu.group.duplicate": "Duplicar", + "menu.group.sort": "Ordenar", + "menu.group.resolve": "Resolver", + "menu.texture.face": "Aplicar a Face", + "menu.texture.cube": "Aplicar aos Cubes", + "menu.texture.file": "Arquivo", + "menu.texture.refresh": "Atualizar", + "menu.texture.change": "Mudar Arquivo", + "menu.texture.folder": "Abrir na Pasta", + "menu.texture.edit": "Editar", + "menu.texture.export": "Salvar como", + "menu.texture.save": "Salvar", + "menu.texture.properties": "Propriedades...", + "menu.preview.background": "Fundo", + "menu.preview.background.load": "Carregar", + "menu.preview.background.position": "Posição", + "menu.preview.background.lock": "Travar a câmera", + "menu.preview.background.remove": "Remover", + "menu.preview.screenshot": "Captura de Tela", + "menu.preview.perspective": "Perspectiva", + "menu.preview.perspective.normal": "Normal", + "menu.preview.quadview": "Visão Quad", + "menu.preview.fullview": "Visão Completa", + "menu.preview.stop_drag": "Pare o posicionamento em segundo plano", + "menu.uv.mapping": "Mapeamento UV", + "menu.uv.mapping.export": "Exportar", + "menu.uv.mapping.rotation": "Rotação", + "menu.uv.mapping.mirror_x": "Espelhar X", + "menu.uv.mapping.mirror_y": "Espelhar Y", + "menu.uv.tint": "Tom", + "menu.uv.texture": "Textura", + "cube.color.light_blue": "Azul claro", + "cube.color.yellow": "Amarelo", + "cube.color.orange": "Laranja", + "cube.color.red": "Vermelho", + "cube.color.purple": "Roxo", + "cube.color.blue": "Azul", + "cube.color.green": "Verde", + "cube.color.lime": "Lime", + "switches.visibility": "Visibilidade", + "switches.export": "Exportar", + "switches.shading": "Sombra", + "switches.autouv": "Auto UV", + "panel.uv": "UV", + "panel.display": "Padrão", + "panel.textures": "Texturas", + "panel.outliner": "Delineador", + "panel.options": "Rotação", + "panel.options.angle": "Ângulo", + "uv_editor.title": "Editor de UV", + "uv_editor.all_faces": "Todas", + "uv_editor.no_faces": "Nenhum", + "face.north": "Norte", + "face.south": "Sul", + "face.west": "Oeste", + "face.east": "Leste", + "face.up": "Cima", + "face.down": "Baixo", + "direction.north": "Norte", + "direction.south": "Sul", + "direction.west": "Oeste", + "direction.east": "Leste", + "direction.top": "Topo", + "direction.bottom": "Baixo", + "display.slot.third_right": "Direito da terceira pessoa", + "display.slot.third_left": "Esquerdo da terceira pessoa", + "display.slot.first_right": "Direito da primeira pessoa", + "display.slot.first_left": "Esquerdo da primeira pessoa", + "display.slot.head": "Cabeça", + "display.slot.ground": "Ground", + "display.slot.frame": "Quadro", + "display.slot.gui": "GUI (Interface)", + "display.rotation": "Rotação", + "display.translation": "Translação", + "display.scale": "Escala", + "display.slot": "Slot", + "display.reference": "Modelo Referência", + "display.presetname": "Nome", + "display.reference.player": "Player", + "display.reference.zombie": "Zumbi", + "display.reference.armor_stand": "Suporte de armadura", + "display.reference.baby_zombie": "Zumbi Bebê", + "display.reference.armor_stand_small": "Suporte de Armadura pequeno", + "display.reference.monitor": "Normal", + "display.reference.bow": "Arco", + "display.reference.block": "Bloco", + "display.reference.frame": "Moldura", + "display.reference.inventory_nine": "3x3", + "display.reference.inventory_full": "Inventário", + "display.reference.hud": "HUD", + "display.preset.blank_name": "Por favor insira um nome", + "display.preset.item": "Item Padrão", + "display.preset.block": "Bloco Padrão", + "display.preset.handheld": "Arma Padrão", + "display.preset.rod": "Vara Padrão", + "dialog.continue": "Continuar", + "message.square_textures": "Texturas têm que ser quadradas", + "message.unsaved_texture.title": "Textura não salva", + "message.unsaved_texture.message": "Todas as alterações não salvas serão perdidas.Você quer continuar?", + "dialog.update.no_connection": "Sem conexão com a Internet", + "dialog.update.latest": "Última versão", + "dialog.update.installed": "Versão Instalada", + "dialog.update.update": "Atualização", + "action.brush_mode.brush": "Modo Volta", + "action.brush_mode.noise": "Modo Nariz", + "action.vertex_snap_mode.move": "Mover", + "action.vertex_snap_mode.scale": "Escala", + "action.open_model_folder": "Abrir pasta de Modelo", + "action.open_model_folder.desc": "Abre a pasta onde o modelo está contido", + "action.change_textures_folder": "Mudar local da textura", + "action.change_textures_folder.desc": "Alterar a pasta em que todas as texturas são salvas", + "menu.texture.particle": "Usar para partículas", + "message.update_notification.title": "Uma atualização está disponível", + "message.update_notification.message": "A versão \"%0\" do Blockbench está disponível.\nVocê deseja instalar agora?", + "message.update_notification.install": "Instalar", + "message.update_notification.later": "Mais tarde", + "message.untextured": "A superfície não tem textura", + "dialog.toolbar_edit.title": "Customizar Barra de Ferramentas", + "dialog.shift_uv.title": "Shift UV", + "dialog.shift_uv.message": "Digite o número que você deseja multiplicar as coordenadas de deslocamento UV por. Expressões matemáticas são permitidas. Prefira um \"+\" se você quiser adicionar esse número.", + "dialog.shift_uv.horizontal": "Horizontal", + "dialog.shift_uv.vertical": "Vertical", + "keybindings.reset": "Resetar", + "keybindings.clear": "Vazio", + "action.cube_counter": "Contador de cubo", + "action.uv_rotation": "Rotação UV", + "action.uv_rotation.desc": "Rotação da face UV", + "action.uv_grid": "Grade UV", + "action.uv_grid.desc": "A resolução da grade na qual o seletor UV se encaixa", + "action.uv_grid.auto": "Auto", + "action.uv_grid.none": "Nenhum", + "action.uv_maximize": "Maximizar UV", + "action.uv_maximize.desc": "Define o UV para este face para a textura total", + "action.uv_auto": "Auto UV", + "action.uv_auto.desc": "Define o tamanho de UV dessa face para o tamanho real da face", + "action.uv_rel_auto": "Rel. Auto UV", + "action.uv_rel_auto.desc": "Define o UV dessa face para a posição e tamanho da face real", + "action.uv_mirror_x": "Espelhar UV X", + "action.uv_mirror_x.desc": "Espelha o UV da face no eixo X", + "action.uv_mirror_y": "Espelhar UV Y", + "action.uv_mirror_y.desc": "Espelha o UV da face no eixo Y", + "action.uv_transparent": "Face Transparente", + "action.uv_transparent.desc": "Faz a face atual, transparente", + "action.uv_reset": "Resetar Face", + "action.uv_reset.desc": "Reseta a Face atual", + "action.cullface": "Face Cobrida", + "action.cullface.desc": "Desativa a renderização para essa face se o lado selecionado do modelo for coberto", + "action.auto_cullface": "Auto Face Cobrida", + "action.auto_cullface.desc": "Define o rosto para este rosto para si", + "action.face_tint": "Tom", + "action.face_tint.desc": "Ativa a opção de tom para a face atual", + "action.uv_shift": "Shift UV", + "action.uv_shift.desc": "Desloca todas as regiões UV por uma quantidade fixa ou expressão matemática", + "menu.toolbar.edit": "Customizar", + "menu.toolbar.reset": "Resetar", + "uv_editor.rotated": "Rotacionado", + "uv_editor.auto_cull": "Face Cobrida para si mesmo", + "uv_editor.copied": "Face Copiada", + "uv_editor.pasted": "Face Colada", + "uv_editor.copied_x": "Copiada %0 Faces ", + "uv_editor.reset": "Resetar Face", + "uv_editor.maximized": "Maximizado", + "uv_editor.autouv": "Tamanho Automático", + "uv_editor.mirrored": "Espelhado", + "uv_editor.to_all": "Aplicado para todas as Faces", + "uv_editor.transparent": "Feito Transparente", + "uv_editor.cullface_on": "Face Cobrida Ligada", + "uv_editor.cullface_off": "Face Cobrida Desligada", + "uv_editor.tint_on": "Tom Ligado", + "uv_editor.tint_off": "Tom Desligado", + "action.uv_apply_all": "Aplicar para todas as Faces", + "action.uv_apply_all.desc": "Aplica as configurações na face atual e para todas as faces", + "message.convert_mode.title": "Converter Modelo", + "message.convert_mode.message": "Você tem certeza que deseja converter esse modelo para %0? Você não poderá desfazer isso.", + "message.convert_mode.block": "um modelo de entidade", + "message.convert_mode.entity": "um modelo de bloco", + "message.convert_mode.convert": "Converter", + "message.image_editor_missing.title": "Editor de Imagem Padrão", + "message.image_editor_missing.message": "Selecione um arquivo executável do seu Editor de Imagem", + "message.image_editor_missing.detail": "O Blockbench não conseguiu encontrar um editor de imagens no seu computador. Selecione o arquivo executável do seu editor de imagens preferido.", + "action.update_autouv": "Atualizar Auto UV", + "action.update_autouv.desc": "Atualiza o auto UV mapeando os cubos selecionados", + "category.uv": "UV", + "status_bar.saved": "O Modelo foi salvo", + "status_bar.unsaved": "Existem alterações não salvas!", + "action.move_up": "Move para cima", + "action.move_up.desc": "Mova os cubos selecionados para cima em relação ao ângulo atual da câmera", + "action.move_down": "Mover para baixo", + "action.move_down.desc": "Mova os cubos selecionados para baixo em relação ao ângulo atual da câmera", + "action.move_left": "Mover para esquerda", + "action.move_left.desc": "Mova os cubos selecionados para a esquerda em relação ao ângulo atual da câmera", + "action.move_right": "Mover para direita", + "action.move_right.desc": "Mova os cubos selecionados para a direita em relação ao ângulo atual da câmera", + "action.move_forth": "Ir para Frente", + "action.move_forth.desc": "Mova os cubos selecionados para a frente em relação ao ângulo atual da câmera", + "action.move_back": "Ir para Trás", + "action.move_back.desc": "Mova os cubos selecionados para a trás em relação ao ângulo atual da câmera", + "layout.color.wireframe": "Wireframe", + "layout.color.wireframe.desc": "Linhas de visualização de wireframe", + "action.add_animation": "Adicionar Animação", + "action.add_animation.desc": "Criar uma animação em branco", + "action.load_animation_file": "Importar Animações", + "action.load_animation_file.desc": "Importar um arquivo de animação", + "action.play_animation": "Iniciar Animação", + "action.play_animation.desc": "Pré Visualização de animação", + "action.export_animation_file": "Exportar Animação", + "action.export_animation_file.desc": "Exporta um arquivo json com as animações atuais", + "action.slider_keyframe_time": "Tempo de Código", + "action.slider_keyframe_time.desc": "Alterar o timecode dos quadros-chave selecionados", + "timeline.rotation": "Rotação", + "timeline.position": "Posição", + "timeline.scale": "Escala", + "menu.timeline.add": "Adicionar quadro-chave", + "menu.keyframe.quaternion": "Quartenion", + "panel.animations": "Anim", + "panel.keyframe": "Quadro - Chave", + "panel.keyframe.type": "Quadro - Chave (%0)", + "generic.delete": "Deletar", + "generic.rename": "Renomear", + "message.rename_animation": "Renomear Animação", + "message.animation_update_var": "Atualização de Variável de Animação", + "message.no_animation_selected": "Você tem que selecionar uma animação para fazer isso", + "message.no_bone_selected": "Você tem que selecionar um osso para fazer isso", + "message.duplicate_groups.title": "Nome do Osso Duplicado", + "message.duplicate_groups.message": "O nome deste osso existe em vários ossos. Isso pode causar problemas.", + "action.select_all_keyframes": "Selecionar todos os Quadros-Chave", + "action.select_all_keyframes.desc": "Seleciona todos os Quadros-Chave do osso atual", + "action.delete_keyframes": "Deletar Quadros-Chave", + "action.delete_keyframes.desc": "Deleta todos os Quadros-Chave selecionados", + "menu.animation": "Animação", + "menu.animation.loop": "Loop", + "menu.animation.override": "Sobrepor", + "menu.animation.anim_time_update": "Atualizar Variável", + "message.display_skin_model.title": "Modelo da Skin", + "message.display_skin_model.message": "Seleciona o tipo de modelo da sua skin", + "message.display_skin_model.classic": "Clássico", + "message.display_skin_model.slim": "Magro/Fino", + "message.bone_material": "Mudar o material do osso", + "action.slider_animation_length": "Duração da animação", + "action.slider_animation_length.desc": "Muda a duração da animação selecionada", + "menu.group.material": "Definir Material", + "action.camera_reset": "Resetar Câmera", + "action.camera_reset.desc": "Redefinir a visualização atual para o ângulo de câmera padrão", + "panel.variable_placeholders": "Espaços reservados variáveis", + "panel.variable_placeholders.info": "Listar as variáveis que você deseja visualizar via nome = valor", + "status_bar.vertex_distance": "Distância: %0", + "dialog.create_gif.title": "Gravar GIF", + "dialog.create_gif.length": "Duração (Segundos)", + "dialog.create_gif.fps": "FPS", + "dialog.create_gif.compression": "Quantidade de Compressão", + "dialog.create_gif.play": "Iniciar Animação", + "category.animation": "Animação", + "action.record_model_gif": "Gravar GIF", + "action.record_model_gif.desc": "Grava um GIF animado do modelo do ângulo atual", + "display.mirror": "Espelhar", + "data.separator": "Separador", + "message.set_background_position.title": "Posição de Fundo", + "menu.preview.background.set_position": "Definir Posição", + "dialog.toolbar_edit.hidden": "Esconder", + "action.export_class_entity": "Exportar Entidade Java", + "action.export_class_entity.desc": "Exporta o modelo da entidade como Java class", + "settings.seethrough_outline": "Esboços de raio X", + "settings.seethrough_outline.desc": "Mostrar contornos através de objetos", + "mode.edit": "Editar", + "mode.paint": "Pintar", + "mode.display": "Padrão", + "mode.animate": "Animar ", + "status_bar.recording_gif": "Gravando GIF", + "status_bar.processing_gif": "Processando GIF ", + "settings.backup_retain": "Backup reter a duração", + "settings.backup_retain.desc": "Defina por quanto tempo o Blockbench mantém backups antigos em dias", + "action.rotate_tool": "Rodar", + "action.rotate_tool.desc": "Ferramenta para selecionar e rodar elementos", + "action.fill_tool": "Lata de Tinda", + "action.fill_tool.desc": "Lata de Tinta para preencher faces inteiras com uma cor", + "action.eraser": "Borracha", + "action.eraser.desc": "Ferramenta Borracha para substituir cores em uma textura com transparência", + "action.color_picker": "Seletor de cores", + "action.color_picker.desc": "Ferramenta para escolher a cor dos pixels na sua textura", + "action.open_backup_folder": "Abrir pasta de Backup", + "action.open_backup_folder.desc": "Abre a pasta de Backup do Blockbench", + "switches.mirror": "Espelhar UV", + "language_name": "Inglês", + "message.plugin_reload": "Recarregado %0 plug-ins locais", + "settings.brightness": "Brilho", + "settings.brightness.desc": "Brilho da pré visualização. Padrão é 50", + "menu.preview.perspective.reset": "Resetar Câmera", + "action.fill_mode": "Modo de preenchimento", + "action.fill_mode.desc": "Modo da ferramenta de preenchimento", + "action.fill_mode.face": "Face", + "action.fill_mode.color": "Cor", + "action.fill_mode.cube": "Cubo", + "action.toggle_mirror_uv": "Espelhar UV", + "action.toggle_mirror_uv.desc": "Alterne o espelhamento UV no eixo X dos cubos selecionados.", + "action.toggle_uv_overlay": "Alternar sobreposição de UV", + "action.toggle_uv_overlay.desc": "Quando ativado, exibe todas as sobreposições de mapeamento UV acima da textura.", + "menu.texture.blank": "Aplicar a faces não texturizadas", + "dialog.scale.select_overflow": "Selecione Overflow", + "dialog.create_texture.compress": "Comprimir Modelo", + "action.action_control": "Controle de ação", + "action.action_control.desc": "Pesquise e execute qualquer ação disponível", + "keybindings.recording": "Gravando de Teclas de Gravação", + "keybindings.press": "Pressione uma tecla ou combinação de teclas ou clique em qualquer lugar da tela para gravar sua combinação de teclas.", + "action.pivot_tool": "Ferramenta Pivô", + "action.pivot_tool.desc": "Ferramenta para alterar o ponto de articulação de cubos e ossos", + "action.slider_animation_speed": "Velocidade de reprodução", + "action.slider_animation_speed.desc": "Velocidade de reprodução da linha do tempo em porcentagem", + "action.previous_keyframe": "Quadro-Anterior anterior", + "action.previous_keyframe.desc": "Pula para o Quadro-Chave Anterior", + "action.next_keyframe": "Próximo Quadro-Chave", + "action.next_keyframe.desc": "Pula para o próximo Quadro-Chave", + "message.outdated_client.title": "Cliente desatualizado", + "message.outdated_client.message": "Por favor atualize o Blockbench para a última versão para fazer isso.", + "action.export_bbmodel": "Exportar Projeto Blockbench", + "action.export_bbmodel.desc": "Exporta um Projeto do Blockbench com todos os cubos, texturas e animações", + "action.export_asset_archive": "Baixar Arquivo", + "action.export_asset_archive.desc": "Baixa um arquivo com o modelo e todas as texturas nele", + "action.upload_sketchfab": "Upload do Sketchfab", + "message.sketchfab.name_or_token": "Por favor insira seu token e nome do Sketchfab", + "dialog.sketchfab_uploader.title": "Upload do modelo Sketchfab", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "O token é usado para conectar o Blockbench à sua conta do Sketchfab. Você pode encontrá-lo em sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Nome do Modelo", + "dialog.sketchfab_uploader.description": "Descrição", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token para autorizar o Blockbench a fazer upload para sua conta do Sketchfab", + "panel.color": "Cor", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" +} \ No newline at end of file diff --git a/lang/ru.json b/lang/ru.json index b85decd0..a662fa40 100644 --- a/lang/ru.json +++ b/lang/ru.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Смещение центра поворота по оси Z", "action.brush_mode": "Режим кисти", "action.brush_mode.desc": "Режим кисти", - "action.brush_color": "Цвет", - "action.brush_color.desc": "Цвет кисти", "action.slider_brush_size": "Размер", "action.slider_brush_size.desc": "Размер кисти в пикселях", "action.slider_brush_opacity": "Непрозрачность", @@ -586,7 +584,6 @@ "panel.outliner": "Элементы", "panel.options": "Поворот", "panel.options.angle": "Угол", - "panel.options.origin": "Центр поворота", "uv_editor.title": "Редактор UV", "uv_editor.all_faces": "Все", "uv_editor.no_faces": "Нет", @@ -823,10 +820,10 @@ "language_name": "Английский", "message.plugin_reload": "Перезагружено %0 локальных плагинов", "settings.brightness": "Яркость", - "settings.brightness.desc": "Brightness of the preview. Default is 50", - "menu.preview.perspective.reset": "Reset Camera", + "settings.brightness.desc": "Яркость дисплея. 50 по умолчанию", + "menu.preview.perspective.reset": "Сбросить камеру", "action.fill_mode": "Режим заполнения", - "action.fill_mode.desc": "Mode of the fill tool", + "action.fill_mode.desc": "Режим инструмента заполнения", "action.fill_mode.face": "Грань", "action.fill_mode.color": "Цвет", "action.fill_mode.cube": "Куб", @@ -854,5 +851,53 @@ "action.export_bbmodel": "Export Blockbench Project", "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.export_asset_archive.desc": "Download an archive with the model and all textures in it", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Тэги", + "settings.sketchfab_token": "Ключ Скетчфаб", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Цвет", + "data.origin": "Центр поворота", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Цвета кубов", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "Файл не найден", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/sv.json b/lang/sv.json index 212d1df6..80c5e376 100644 --- a/lang/sv.json +++ b/lang/sv.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "Flytta ursprunget på Z axeln", "action.brush_mode": "Penselläge", "action.brush_mode.desc": "Penselns läge", - "action.brush_color": "Färg", - "action.brush_color.desc": "Färg på penseln", "action.slider_brush_size": "Storlek", "action.slider_brush_size.desc": "Penselns radie i pixlar", "action.slider_brush_opacity": "Opacitet", @@ -586,7 +584,6 @@ "panel.outliner": "Konturen", "panel.options": "Rotation", "panel.options.angle": "Vinkel", - "panel.options.origin": "Ursprung", "uv_editor.title": "UV redigerare", "uv_editor.all_faces": "Alla", "uv_editor.no_faces": "Ingen", @@ -854,5 +851,53 @@ "action.export_bbmodel": "Export Blockbench Project", "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "action.export_asset_archive.desc": "Download an archive with the model and all textures in it", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Color", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model", + "message.recover_backup.title": "Recover Model", + "message.recover_backup.message": "Blockbench was closed without saving. Do you want to recover the model?", + "message.install_plugin": "Installing the plugin %0", + "message.invalid_session.title": "Invalid Session Token", + "message.invalid_session.message": "The session you are trying to join has expired or the token provided is invalid.", + "dialog.create_texture.power": "Power-of-2 Size", + "dialog.create_gif.turn": "Turntable Speed", + "action.edit_session": "Edit Session...", + "action.edit_session.desc": "Connect to an edit session to collaborate with other users", + "action.reset_keyframe": "Reset Keyframe", + "action.reset_keyframe.desc": "Reset all values of the selected keyframes", + "panel.options.origin": "Origin", + "dialog.edit_session.title": "Edit Session", + "edit_session.username": "Username", + "edit_session.token": "Token", + "edit_session.about": "Edit Sessions can be used to collaborate on models across the internet. Create a session and copy the token and send it to friends, who can then use it to join.", + "edit_session.join": "Join Session", + "edit_session.create": "Create Session", + "edit_session.quit": "Quit Session", + "edit_session.joined": "User %0 joined the session", + "edit_session.left": "User %0 left the session", + "edit_session.quit_session": "Left current session", + "edit_session.status": "Status", + "edit_session.hosting": "Hosting", + "edit_session.connected": "Connected to a session" } \ No newline at end of file diff --git a/lang/zh.json b/lang/zh.json index bca3656f..c5a62d0e 100644 --- a/lang/zh.json +++ b/lang/zh.json @@ -326,8 +326,6 @@ "action.slider_origin_z.desc": "以 Z 轴移动原点", "action.brush_mode": "笔刷模式", "action.brush_mode.desc": "笔刷的模式", - "action.brush_color": "颜色", - "action.brush_color.desc": "笔刷的颜色", "action.slider_brush_size": "尺寸", "action.slider_brush_size.desc": "笔刷的半径(以像素为单位)", "action.slider_brush_opacity": "不透明度", @@ -420,7 +418,7 @@ "action.add_group.desc": "添加一个新的组", "action.outliner_toggle": "切换更多选项", "action.outliner_toggle.desc": "切换开关以在outliner中添加更多选项", - "action.duplicate": "复制", + "action.duplicate": "生成副本", "action.duplicate.desc": "复制选定的方块或组", "action.delete": "删除", "action.delete.desc": "删除选定的方块或组", @@ -532,12 +530,12 @@ "menu.view.zoom": "缩放", "menu.view.background": "背景", "menu.view.screenshot": "截图", - "menu.cube.duplicate": "复制", + "menu.cube.duplicate": "生成副本", "menu.cube.color": "标记颜色", "menu.cube.texture": "材质纹理", "menu.cube.texture.transparent": "透明度", "menu.cube.texture.blank": "空白", - "menu.group.duplicate": "复制", + "menu.group.duplicate": "生成副本", "menu.group.sort": "排序", "menu.group.resolve": "解析", "menu.texture.face": "应用到面", @@ -586,7 +584,6 @@ "panel.outliner": "大纲", "panel.options": "旋转", "panel.options.angle": "角度", - "panel.options.origin": "旋转原点", "uv_editor.title": "UV 编辑器", "uv_editor.all_faces": "全部", "uv_editor.no_faces": "无", @@ -839,20 +836,43 @@ "dialog.create_texture.compress": "压缩模板", "action.action_control": "动作控制", "action.action_control.desc": "搜索并且执行任何可用的操作", - "keybindings.recording": "Recording Keybinding", - "keybindings.press": "Press a key or key combination or click anywhere on the screen to record your keybinding.", - "action.pivot_tool": "Pivot Tool", - "action.pivot_tool.desc": "Tool to change the pivot point of cubes and bones", - "action.slider_animation_speed": "Playback Speed", - "action.slider_animation_speed.desc": "Playback speed of the timeline in percent", - "action.previous_keyframe": "Previous Keyframe", - "action.previous_keyframe.desc": "Jump to the previous keyframe", - "action.next_keyframe": "Next Keyframe", - "action.next_keyframe.desc": "Jump to the next keyframe", - "message.outdated_client.title": "Outdated client", - "message.outdated_client.message": "Please update to the latest version of Blockbench to do this.", - "action.export_bbmodel": "Export Blockbench Project", - "action.export_bbmodel.desc": "Export a Blockbench project with all cubes, textures and animations", - "action.export_asset_archive": "Download Archive", - "action.export_asset_archive.desc": "Download an archive with the model and all textures in it" + "keybindings.recording": "录制按键绑定", + "keybindings.press": "输入按键或输入组合按键或单击屏幕上的任意位置以记录键绑定", + "action.pivot_tool": "枢轴工具", + "action.pivot_tool.desc": "用于更改立方体和骨骼的轴心点的工具", + "action.slider_animation_speed": "播放速度", + "action.slider_animation_speed.desc": "时间线的播放速度以百分比表示", + "action.previous_keyframe": "上一个关键帧", + "action.previous_keyframe.desc": "跳转到上一个关键帧", + "action.next_keyframe": "下一个关键帧", + "action.next_keyframe.desc": "跳转到下一个关键帧", + "message.outdated_client.title": "Blockbench已经过期(请更新)", + "message.outdated_client.message": "请更新到Blockbench的最新版本来执行此操作", + "action.export_bbmodel": "导出Blockbench项目", + "action.export_bbmodel.desc": "导出包含所有立方体,纹理和动画的Blockbench项目", + "action.export_asset_archive": "下载存档", + "action.export_asset_archive.desc": "下载包含模型及其中所有纹理的存档", + "action.upload_sketchfab": "Sketchfab Upload", + "message.sketchfab.name_or_token": "Please enter your Sketchfab token and a name", + "dialog.sketchfab_uploader.title": "Upload Sketchfab Model", + "dialog.sketchfab_uploader.token": "API Token", + "dialog.sketchfab_uploader.about_token": "The token is used to connect Blockbench to your Sketchfab account. You can find it on sketchfab.com/settings/password", + "dialog.sketchfab_uploader.name": "Model Name", + "dialog.sketchfab_uploader.description": "Description", + "dialog.sketchfab_uploader.tags": "Tags", + "settings.sketchfab_token": "Sketchfab Token", + "settings.sketchfab_token.desc": "Token to authorize Blockbench to upload to your Sketchfab account", + "panel.color": "Color", + "data.origin": "Origin", + "message.sketchfab.success": "Uploaded model successfully", + "message.sketchfab.error": "Failed to upload model to Sketchfab", + "settings.outliner_colors": "Outliner Colors", + "settings.outliner_colors.desc": "Display cube colors in the outliner", + "action.upload_sketchfab.desc": "Upload your model to Sketchfab", + "action.element_colors": "Cube Colors", + "action.element_colors.desc": "Show cube colors in the outliner", + "texture.error.file": "File not found", + "texture.error.invalid": "Invalid file", + "texture.error.ratio": "Invalid aspect ratio", + "texture.error.parent": "Texture file provided by parent model" } \ No newline at end of file diff --git a/lib/peer.min.js b/lib/peer.min.js new file mode 100644 index 00000000..03365756 --- /dev/null +++ b/lib/peer.min.js @@ -0,0 +1 @@ +!function o(s,a,u){function c(t,e){if(!a[t]){if(!s[t]){var i="function"==typeof require&&require;if(!e&&i)return i(t,!0);if(p)return p(t,!0);var n=new Error("Cannot find module '"+t+"'");throw n.code="MODULE_NOT_FOUND",n}var r=a[t]={exports:{}};s[t][0].call(r.exports,function(e){return c(s[t][1][e]||e)},r,r.exports,o,s,a,u)}return a[t].exports}for(var p="function"==typeof require&&require,e=0;e<u.length;e++)c(u[e]);return c}({1:[function(e,t,i){"use strict";Object.defineProperty(i,"__esModule",{value:!0}),i.RTCSessionDescription=window.RTCSessionDescription||window.mozRTCSessionDescription,i.RTCPeerConnection=window.RTCPeerConnection||window.mozRTCPeerConnection||window.webkitRTCPeerConnection,i.RTCIceCandidate=window.RTCIceCandidate||window.mozRTCIceCandidate},{}],2:[function(e,t,i){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var a=e("./util"),n=e("eventemitter3"),r=e("./negotiator"),o=e("reliable");function s(e,t,i){if(!(this instanceof s))return new s(e,t,i);n.EventEmitter.call(this),this.options=a.util.extend({serialization:"binary",reliable:!1},i),this.open=!1,this.type="data",this.peer=e,this.provider=t,this.id=this.options.connectionId||s._idPrefix+a.util.randomToken(),this.label=this.options.label||this.id,this.metadata=this.options.metadata,this.serialization=this.options.serialization,this.reliable=this.options.reliable,this._buffer=[],this._buffering=!1,this.bufferSize=0,this._chunkedData={},this.options._payload&&(this._peerBrowser=this.options._payload.browser),r.Negotiator.startConnection(this,this.options._payload||{originator:!0})}i.DataConnection=s,a.util.inherits(s,n.EventEmitter),s._idPrefix="dc_",s.prototype.initialize=function(e){this._dc=this.dataChannel=e,this._configureDataChannel()},s.prototype._configureDataChannel=function(){var t=this;a.util.supports.sctp&&(this._dc.binaryType="arraybuffer"),this._dc.onopen=function(){a.util.log("Data channel connection success"),t.open=!0,t.emit("open")},!a.util.supports.sctp&&this.reliable&&(this._reliable=new o.Reliable(this._dc,a.util.debug)),this._reliable?this._reliable.onmessage=function(e){t.emit("data",e)}:this._dc.onmessage=function(e){t._handleDataMessage(e)},this._dc.onclose=function(e){a.util.log("DataChannel closed for:",t.peer),t.close()}},s.prototype._handleDataMessage=function(e){var t=this,i=e.data,n=i.constructor;if("binary"===this.serialization||"binary-utf8"===this.serialization){if(n===Blob)return void a.util.blobToArrayBuffer(i,function(e){i=a.util.unpack(e),t.emit("data",i)});if(n===ArrayBuffer)i=a.util.unpack(i);else if(n===String){var r=a.util.binaryStringToArrayBuffer(i);i=a.util.unpack(r)}}else"json"===this.serialization&&(i=JSON.parse(i));if(i.__peerData){var o=i.__peerData,s=this._chunkedData[o]||{data:[],count:0,total:i.total};return s.data[i.n]=i.data,s.count+=1,s.total===s.count&&(delete this._chunkedData[o],i=new Blob(s.data),this._handleDataMessage({data:i})),void(this._chunkedData[o]=s)}this.emit("data",i)},s.prototype.close=function(){this.open&&(this.open=!1,r.Negotiator.cleanup(this),this.emit("close"))},s.prototype.send=function(e,t){if(this.open)if(this._reliable)this._reliable.send(e);else{var i=this;if("json"===this.serialization)this._bufferedSend(JSON.stringify(e));else if("binary"===this.serialization||"binary-utf8"===this.serialization){var n=a.util.pack(e);if((a.util.chunkedBrowsers[this._peerBrowser]||a.util.chunkedBrowsers[a.util.browser])&&!t&&n.size>a.util.chunkedMTU)return void this._sendChunks(n);a.util.supports.sctp?a.util.supports.binaryBlob?this._bufferedSend(n):a.util.blobToArrayBuffer(n,function(e){i._bufferedSend(e)}):a.util.blobToBinaryString(n,function(e){i._bufferedSend(e)})}else this._bufferedSend(e)}else this.emit("error",new Error("Connection is not open. You should listen for the `open` event before sending messages."))},s.prototype._bufferedSend=function(e){!this._buffering&&this._trySend(e)||(this._buffer.push(e),this.bufferSize=this._buffer.length)},s.prototype._trySend=function(e){try{this._dc.send(e)}catch(e){this._buffering=!0;var t=this;return setTimeout(function(){t._buffering=!1,t._tryBuffer()},100),!1}return!0},s.prototype._tryBuffer=function(){if(0!==this._buffer.length){var e=this._buffer[0];this._trySend(e)&&(this._buffer.shift(),this.bufferSize=this._buffer.length,this._tryBuffer())}},s.prototype._sendChunks=function(e){for(var t=a.util.chunk(e),i=0,n=t.length;i<n;i+=1){e=t[i];this.send(e,!0)}},s.prototype.handleMessage=function(e){var t=e.payload;switch(e.type){case"ANSWER":this._peerBrowser=t.browser,r.Negotiator.handleSDP(e.type,this,t.sdp);break;case"CANDIDATE":r.Negotiator.handleCandidate(this,t.candidate);break;default:a.util.warn("Unrecognized message type:",e.type,"from peer:",this.peer)}}},{"./negotiator":5,"./util":8,eventemitter3:9,reliable:12}],3:[function(e,t,i){"use strict";var n=this&&this.__importDefault||function(e){return e&&e.__esModule?e:{default:e}};Object.defineProperty(i,"__esModule",{value:!0});var r=e("./util"),o=e("./adapter"),s=e("./socket"),a=e("./mediaconnection"),u=e("./dataconnection"),c=e("./peer"),p=e("./negotiator"),l=n(e("js-binarypack"));window.Socket=s.Socket,window.MediaConnection=a.MediaConnection,window.DataConnection=u.DataConnection,window.Peer=c.Peer,window.RTCPeerConnection=o.RTCPeerConnection,window.RTCSessionDescription=o.RTCSessionDescription,window.RTCIceCandidate=o.RTCIceCandidate,window.Negotiator=p.Negotiator,window.util=r.util,window.BinaryPack=l.default},{"./adapter":1,"./dataconnection":2,"./mediaconnection":4,"./negotiator":5,"./peer":6,"./socket":7,"./util":8,"js-binarypack":10}],4:[function(e,t,i){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var r=e("./util"),n=e("eventemitter3"),o=e("./negotiator");function s(e,t,i){if(!(this instanceof s))return new s(e,t,i);n.EventEmitter.call(this),this.options=r.util.extend({},i),this.open=!1,this.type="media",this.peer=e,this.provider=t,this.metadata=this.options.metadata,this.localStream=this.options._stream,this.id=this.options.connectionId||s._idPrefix+r.util.randomToken(),this.localStream&&o.Negotiator.startConnection(this,{_stream:this.localStream,originator:!0})}i.MediaConnection=s,r.util.inherits(s,n.EventEmitter),s._idPrefix="mc_",s.prototype.addStream=function(e){r.util.log("Receiving stream",e),this.remoteStream=e,this.emit("stream",e)},s.prototype.handleMessage=function(e){var t=e.payload;switch(e.type){case"ANSWER":o.Negotiator.handleSDP(e.type,this,t.sdp),this.open=!0;break;case"CANDIDATE":o.Negotiator.handleCandidate(this,t.candidate);break;default:r.util.warn("Unrecognized message type:",e.type,"from peer:",this.peer)}},s.prototype.answer=function(e){if(this.localStream)r.util.warn("Local stream already exists on this MediaConnection. Are you answering a call twice?");else{this.options._payload._stream=e,this.localStream=e,o.Negotiator.startConnection(this,this.options._payload);for(var t=this.provider._getMessages(this.id),i=0,n=t.length;i<n;i+=1)this.handleMessage(t[i]);this.open=!0}},s.prototype.close=function(){this.open&&(this.open=!1,o.Negotiator.cleanup(this),this.emit("close"))}},{"./negotiator":5,"./util":8,eventemitter3:9}],5:[function(e,t,o){"use strict";Object.defineProperty(o,"__esModule",{value:!0});var s=e("./util"),r=e("./adapter");o.Negotiator={pcs:{data:{},media:{}},queue:[]},o.Negotiator._idPrefix="pc_",o.Negotiator.startConnection=function(e,t){var i=o.Negotiator._getPeerConnection(e,t);if(e.pc=e.peerConnection=i,"media"===e.type&&t._stream&&i.addStream(t._stream),t.originator){if("data"===e.type){var n={};s.util.supports.sctp||(n={reliable:t.reliable});var r=i.createDataChannel(e.label,n);e.initialize(r)}o.Negotiator._makeOffer(e)}else o.Negotiator.handleSDP("OFFER",e,t.sdp)},o.Negotiator._getPeerConnection=function(e,t){o.Negotiator.pcs[e.type]||s.util.error(e.type+" is not a valid connection type. Maybe you overrode the `type` property somewhere."),o.Negotiator.pcs[e.type][e.peer]||(o.Negotiator.pcs[e.type][e.peer]={});var i;o.Negotiator.pcs[e.type][e.peer];return t.pc&&(i=o.Negotiator.pcs[e.type][e.peer][t.pc]),i&&"stable"===i.signalingState||(i=o.Negotiator._startPeerConnection(e)),i},o.Negotiator._startPeerConnection=function(e){s.util.log("Creating RTCPeerConnection.");var t=o.Negotiator._idPrefix+s.util.randomToken(),i={};"data"!==e.type||s.util.supports.sctp?"media"===e.type&&(i={optional:[{DtlsSrtpKeyAgreement:!0}]}):i={optional:[{RtpDataChannels:!0}]};var n=new r.RTCPeerConnection(e.provider.options.config,i);return o.Negotiator.pcs[e.type][e.peer][t]=n,o.Negotiator._setupListeners(e,n,t),n},o.Negotiator._setupListeners=function(t,e,i){var n=t.peer,r=t.id,o=t.provider;s.util.log("Listening for ICE candidates."),e.onicecandidate=function(e){e.candidate&&(s.util.log("Received ICE candidates for:",t.peer),o.socket.send({type:"CANDIDATE",payload:{candidate:e.candidate,type:t.type,connectionId:t.id},dst:n}))},e.oniceconnectionstatechange=function(){switch(e.iceConnectionState){case"failed":s.util.log("iceConnectionState is disconnected, closing connections to "+n),t.emit("error",new Error("Negotiation of connection to "+n+" failed.")),t.close();break;case"disconnected":s.util.log("iceConnectionState is disconnected, closing connections to "+n),t.close();break;case"completed":e.onicecandidate=s.util.noop}},e.onicechange=e.oniceconnectionstatechange,s.util.log("Listening for data channel"),e.ondatachannel=function(e){s.util.log("Received data channel");var t=e.channel;o.getConnection(n,r).initialize(t)},s.util.log("Listening for remote stream"),e.ontrack=function(e){s.util.log("Received remote stream");var t=e.streams[0],i=o.getConnection(n,r);"media"===i.type&&i.addStream(t)}},o.Negotiator.cleanup=function(e){s.util.log("Cleaning up PeerConnection to "+e.peer);var t=e.pc;t&&(t.readyState&&"closed"!==t.readyState||"closed"!==t.signalingState)&&(t.close(),e.pc=null)},o.Negotiator._makeOffer=function(t){var i=t.pc;i.createOffer(function(e){s.util.log("Created offer."),!s.util.supports.sctp&&"data"===t.type&&t.reliable&&(e.sdp=Reliable.higherBandwidthSDP(e.sdp)),i.setLocalDescription(e,function(){s.util.log("Set localDescription: offer","for:",t.peer),t.provider.socket.send({type:"OFFER",payload:{sdp:e,type:t.type,label:t.label,connectionId:t.id,reliable:t.reliable,serialization:t.serialization,metadata:t.metadata,browser:s.util.browser},dst:t.peer})},function(e){"OperationError: Failed to set local offer sdp: Called in wrong state: kHaveRemoteOffer"!=e&&(t.provider.emitError("webrtc",e),s.util.log("Failed to setLocalDescription, ",e))})},function(e){t.provider.emitError("webrtc",e),s.util.log("Failed to createOffer, ",e)},t.options.constraints)},o.Negotiator._makeAnswer=function(t){var i=t.pc;i.createAnswer(function(e){s.util.log("Created answer."),!s.util.supports.sctp&&"data"===t.type&&t.reliable&&(e.sdp=Reliable.higherBandwidthSDP(e.sdp)),i.setLocalDescription(e,function(){s.util.log("Set localDescription: answer","for:",t.peer),t.provider.socket.send({type:"ANSWER",payload:{sdp:e,type:t.type,connectionId:t.id,browser:s.util.browser},dst:t.peer})},function(e){t.provider.emitError("webrtc",e),s.util.log("Failed to setLocalDescription, ",e)})},function(e){t.provider.emitError("webrtc",e),s.util.log("Failed to create answer, ",e)})},o.Negotiator.handleSDP=function(e,t,i){i=new r.RTCSessionDescription(i);var n=t.pc;s.util.log("Setting remote description",i),n.setRemoteDescription(i,function(){s.util.log("Set remoteDescription:",e,"for:",t.peer),"OFFER"===e&&o.Negotiator._makeAnswer(t)},function(e){t.provider.emitError("webrtc",e),s.util.log("Failed to setRemoteDescription, ",e)})},o.Negotiator.handleCandidate=function(e,t){var i=t.candidate,n=t.sdpMLineIndex;e.pc.addIceCandidate(new r.RTCIceCandidate({sdpMLineIndex:n,candidate:i})),s.util.log("Added ICE candidate for:",e.peer)}},{"./adapter":1,"./util":8}],6:[function(e,t,i){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var p=e("./util"),n=e("eventemitter3"),r=e("./socket"),l=e("./mediaconnection"),h=e("./dataconnection");function o(e,t){if(!(this instanceof o))return new o(e,t);n.EventEmitter.call(this),e&&e.constructor==Object?(t=e,e=void 0):e&&(e=e.toString()),(t=p.util.extend({debug:0,host:p.util.CLOUD_HOST,port:p.util.CLOUD_PORT,path:"/",token:p.util.randomToken(),config:p.util.defaultConfig},t)).key="peerjs","/"===(this.options=t).host&&(t.host=window.location.hostname),"/"!==t.path[0]&&(t.path="/"+t.path),"/"!==t.path[t.path.length-1]&&(t.path+="/"),void 0===t.secure&&t.host!==p.util.CLOUD_HOST?t.secure=p.util.isSecure():t.host==p.util.CLOUD_HOST&&(t.secure=!0),t.logFunction&&p.util.setLogFunction(t.logFunction),p.util.setLogLevel(t.debug),p.util.supports.audioVideo||p.util.supports.data?p.util.validateId(e)?(this.destroyed=!1,this.disconnected=!1,this.open=!1,this.connections={},this._lostMessages={},this._initializeServerConnection(),e?this._initialize(e):this._retrieveId()):this._delayedAbort("invalid-id",'ID "'+e+'" is invalid'):this._delayedAbort("browser-incompatible","The current browser does not support WebRTC")}i.Peer=o,p.util.inherits(o,n.EventEmitter),o.prototype._initializeServerConnection=function(){var t=this;this.socket=new r.Socket(this.options.secure,this.options.host,this.options.port,this.options.path,this.options.key,this.options.wsport),this.socket.on("message",function(e){t._handleMessage(e)}),this.socket.on("error",function(e){t._abort("socket-error",e)}),this.socket.on("disconnected",function(){t.disconnected||(t.emitError("network","Lost connection to server."),t.disconnect())}),this.socket.on("close",function(){t.disconnected||t._abort("socket-closed","Underlying socket is already closed.")})},o.prototype._retrieveId=function(e){var i=this,t=new XMLHttpRequest,n=(this.options.secure?"https://":"http://")+this.options.host+":"+this.options.port+this.options.path+this.options.key+"/id";n+="?ts="+(new Date).getTime()+Math.random(),t.open("get",n,!0),t.onerror=function(e){p.util.error("Error retrieving ID",e);var t="";"/"===i.options.path&&i.options.host!==p.util.CLOUD_HOST&&(t=" If you passed in a `path` to your self-hosted PeerServer, you'll also need to pass in that same path when creating a new Peer."),i._abort("server-error","Could not get an ID from the server."+t)},t.onreadystatechange=function(){4===t.readyState&&(200===t.status?i._initialize(t.responseText):t.onerror())},t.send(null)},o.prototype._initialize=function(e){this.id=e,this.socket.start(this.id,this.options.token)},o.prototype._handleMessage=function(e){var t,i=e.type,n=e.payload,r=e.src;switch(i){case"OPEN":this.emit("open",this.id),this.open=!0;break;case"ERROR":this._abort("server-error",n.msg);break;case"ID-TAKEN":this._abort("unavailable-id","ID `"+this.id+"` is taken");break;case"INVALID-KEY":this._abort("invalid-key",'API KEY "'+this.options.key+'" is invalid');break;case"LEAVE":p.util.log("Received leave message from",r),this._cleanupPeer(r);break;case"EXPIRE":this.emitError("peer-unavailable","Could not connect to peer "+r);break;case"OFFER":var o=n.connectionId;if((t=this.getConnection(r,o))&&(t.close(),p.util.warn("Offer received for existing Connection ID:",o)),"media"===n.type)t=new l.MediaConnection(r,this,{connectionId:o,_payload:n,metadata:n.metadata}),this._addConnection(r,t),this.emit("call",t);else{if("data"!==n.type)return void p.util.warn("Received malformed connection type:",n.type);t=new h.DataConnection(r,this,{connectionId:o,_payload:n,metadata:n.metadata,label:n.label,serialization:n.serialization,reliable:n.reliable}),this._addConnection(r,t),this.emit("connection",t)}for(var s=this._getMessages(o),a=0,u=s.length;a<u;a+=1)t.handleMessage(s[a]);break;default:if(!n)return void p.util.warn("You received a malformed message from "+r+" of type "+i);var c=n.connectionId;(t=this.getConnection(r,c))&&t.pc?t.handleMessage(e):c?this._storeMessage(c,e):p.util.warn("You received an unrecognized message:",e)}},o.prototype._storeMessage=function(e,t){this._lostMessages[e]||(this._lostMessages[e]=[]),this._lostMessages[e].push(t)},o.prototype._getMessages=function(e){var t=this._lostMessages[e];return t?(delete this._lostMessages[e],t):[]},o.prototype.connect=function(e,t){if(this.disconnected)return p.util.warn("You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect, or call reconnect on this peer if you believe its ID to still be available."),void this.emitError("disconnected","Cannot connect to new Peer after disconnecting from server.");var i=new h.DataConnection(e,this,t);return this._addConnection(e,i),i},o.prototype.call=function(e,t,i){if(this.disconnected)return p.util.warn("You cannot connect to a new Peer because you called .disconnect() on this Peer and ended your connection with the server. You can create a new Peer to reconnect."),void this.emitError("disconnected","Cannot connect to new Peer after disconnecting from server.");if(t){(i=i||{})._stream=t;var n=new l.MediaConnection(e,this,i);return this._addConnection(e,n),n}p.util.error("To call a peer, you must provide a stream from your browser's `getUserMedia`.")},o.prototype._addConnection=function(e,t){this.connections[e]||(this.connections[e]=[]),this.connections[e].push(t)},o.prototype.getConnection=function(e,t){var i=this.connections[e];if(!i)return null;for(var n=0,r=i.length;n<r;n++)if(i[n].id===t)return i[n];return null},o.prototype._delayedAbort=function(e,t){var i=this;p.util.setZeroTimeout(function(){i._abort(e,t)})},o.prototype._abort=function(e,t){p.util.error("Aborting!"),this._lastServerId?this.disconnect():this.destroy(),this.emitError(e,t)},o.prototype.emitError=function(e,t){p.util.error("Error:",t),"string"==typeof t&&(t=new Error(t)),t.type=e,this.emit("error",t)},o.prototype.destroy=function(){this.destroyed||(this._cleanup(),this.disconnect(),this.destroyed=!0)},o.prototype._cleanup=function(){if(this.connections)for(var e=Object.keys(this.connections),t=0,i=e.length;t<i;t++)this._cleanupPeer(e[t]);this.emit("close")},o.prototype._cleanupPeer=function(e){for(var t=this.connections[e],i=0,n=t.length;i<n;i+=1)t[i].close()},o.prototype.disconnect=function(){var e=this;p.util.setZeroTimeout(function(){e.disconnected||(e.disconnected=!0,e.open=!1,e.socket&&e.socket.close(),e.emit("disconnected",e.id),e._lastServerId=e.id,e.id=null)})},o.prototype.reconnect=function(){if(this.disconnected&&!this.destroyed)p.util.log("Attempting reconnection to server with ID "+this._lastServerId),this.disconnected=!1,this._initializeServerConnection(),this._initialize(this._lastServerId);else{if(this.destroyed)throw new Error("This peer cannot reconnect to the server. It has already been destroyed.");if(this.disconnected||this.open)throw new Error("Peer "+this.id+" cannot reconnect because it is not disconnected from the server!");p.util.error("In a hurry? We're still trying to make the initial connection!")}},o.prototype.listAllPeers=function(t){t=t||function(){};var i=this,n=new XMLHttpRequest,e=(this.options.secure?"https://":"http://")+this.options.host+":"+this.options.port+this.options.path+this.options.key+"/peers";e+="?ts="+(new Date).getTime()+Math.random(),n.open("get",e,!0),n.onerror=function(e){i._abort("server-error","Could not get peers from the server."),t([])},n.onreadystatechange=function(){if(4===n.readyState){if(401===n.status){var e="";throw e=i.options.host!==p.util.CLOUD_HOST?"It looks like you're using the cloud server. You can email team@peerjs.com to enable peer listing for your API key.":"You need to enable `allow_discovery` on your self-hosted PeerServer to use this feature.",t([]),new Error("It doesn't look like you have permission to list peers IDs. "+e)}200!==n.status?t([]):t(JSON.parse(n.responseText))}},n.send(null)}},{"./dataconnection":2,"./mediaconnection":4,"./socket":7,"./util":8,eventemitter3:9}],7:[function(e,t,i){"use strict";Object.defineProperty(i,"__esModule",{value:!0});var o=e("./util"),u=e("eventemitter3");function c(e,t,i,n,r,o){if(!(this instanceof c))return new c(e,t,i,n,r,o);o=o||i,u.EventEmitter.call(this),this.disconnected=!1,this._queue=[];var s=e?"https://":"http://",a=e?"wss://":"ws://";this._httpUrl=s+t+":"+i+n+r,this._wsUrl=a+t+":"+o+n+"peerjs?key="+r}i.Socket=c,o.util.inherits(c,u.EventEmitter),c.prototype.start=function(e,t){this.id=e,this._httpUrl+="/"+e+"/"+t,this._wsUrl+="&id="+e+"&token="+t,this._startXhrStream(),this._startWebSocket()},c.prototype._startWebSocket=function(e){var i=this;this._socket||(this._socket=new WebSocket(this._wsUrl),this._socket.onmessage=function(t){try{var e=JSON.parse(t.data)}catch(e){return void o.util.log("Invalid server message",t.data)}i.emit("message",e)},this._socket.onclose=function(e){o.util.log("Socket closed."),i.disconnected=!0,i.emit("disconnected")},this._socket.onopen=function(){i._timeout&&(clearTimeout(i._timeout),setTimeout(function(){i._http.abort(),i._http=null},5e3)),i._sendQueuedMessages(),o.util.log("Socket open")})},c.prototype._startXhrStream=function(e){try{var t=this;this._http=new XMLHttpRequest,this._http._index=1,this._http._streamIndex=e||0,this._http.open("post",this._httpUrl+"/id?i="+this._http._streamIndex,!0),this._http.onerror=function(){clearTimeout(t._timeout),t.emit("disconnected")},this._http.onreadystatechange=function(){2==this.readyState&&this.old?(this.old.abort(),delete this.old):2<this.readyState&&200===this.status&&this.responseText&&t._handleStream(this)},this._http.send(null),this._setHTTPTimeout()}catch(e){o.util.log("XMLHttpRequest not available; defaulting to WebSockets")}},c.prototype._handleStream=function(t){var e=t.responseText.split("\n");if(t._buffer)for(;0<t._buffer.length;){var i=t._buffer.shift(),n=e[i];try{n=JSON.parse(n)}catch(e){t._buffer.shift(i);break}this.emit("message",n)}var r=e[t._index];if(r)if(t._index+=1,t._index===e.length)t._buffer||(t._buffer=[]),t._buffer.push(t._index-1);else{try{r=JSON.parse(r)}catch(e){return void o.util.log("Invalid server message",r)}this.emit("message",r)}},c.prototype._setHTTPTimeout=function(){var t=this;this._timeout=setTimeout(function(){var e=t._http;t._wsOpen()?e.abort():(t._startXhrStream(e._streamIndex+1),t._http.old=e)},25e3)},c.prototype._wsOpen=function(){return this._socket&&1==this._socket.readyState},c.prototype._sendQueuedMessages=function(){for(var e=0,t=this._queue.length;e<t;e+=1)this.send(this._queue[e])},c.prototype.send=function(e){if(!this.disconnected)if(this.id)if(e.type){var t=JSON.stringify(e);if(this._wsOpen())this._socket.send(t);else{var i=new XMLHttpRequest,n=this._httpUrl+"/"+e.type.toLowerCase();i.open("post",n,!0),i.setRequestHeader("Content-Type","application/json"),i.send(t)}}else this.emit("error","Invalid message");else this._queue.push(e)},c.prototype.close=function(){!this.disconnected&&this._wsOpen()&&(this._socket.close(),this.disconnected=!0)}},{"./util":8,eventemitter3:9}],8:[function(e,t,c){"use strict";var i=this&&this.__importDefault||function(e){return e&&e.__esModule?e:{default:e}};Object.defineProperty(c,"__esModule",{value:!0});var u={iceServers:[{urls:"stun:stun.l.google.com:19302"}]},p=1,n=i(e("js-binarypack")),l=e("./adapter");c.util={noop:function(){},CLOUD_HOST:"0.peerjs.com",CLOUD_PORT:443,chunkedBrowsers:{Chrome:1},chunkedMTU:16300,logLevel:0,setLogLevel:function(e){var t=parseInt(e,10);isNaN(parseInt(e,10))?c.util.logLevel=e?3:0:c.util.logLevel=t,c.util.log=c.util.warn=c.util.error=c.util.noop,0<c.util.logLevel&&(c.util.error=c.util._printWith("ERROR")),1<c.util.logLevel&&(c.util.warn=c.util._printWith("WARNING")),2<c.util.logLevel&&(c.util.log=c.util._print)},setLogFunction:function(e){e.constructor!==Function?c.util.warn("The log function you passed in is not a function. Defaulting to regular logs."):c.util._print=e},_printWith:function(t){return function(){var e=Array.prototype.slice.call(arguments);e.unshift(t),c.util._print.apply(c.util,e)}},_print:function(){var e=!1,t=Array.prototype.slice.call(arguments);t.unshift("PeerJS: ");for(var i=0,n=t.length;i<n;i++)t[i]instanceof Error&&(t[i]="("+t[i].name+") "+t[i].message,e=!0);e?console.error.apply(console,t):console.log.apply(console,t)},defaultConfig:u,browser:window.mozRTCPeerConnection?"Firefox":window.webkitRTCPeerConnection?"Chrome":window.RTCPeerConnection?"Supported":"Unsupported",supports:function(){if(void 0===l.RTCPeerConnection)return{};var e,t,i=!0,n=!0,r=!1,o=!1,s=!!window.webkitRTCPeerConnection;try{e=new l.RTCPeerConnection(u,{optional:[{RtpDataChannels:!0}]})}catch(e){n=i=!1}if(i)try{t=e.createDataChannel("_PEERJSTEST")}catch(e){i=!1}if(i){try{t.binaryType="blob",r=!0}catch(e){}var a=new l.RTCPeerConnection(u,{});try{o=a.createDataChannel("_PEERJSRELIABLETEST",{}).reliable}catch(e){}a.close()}return n&&(n=!!e.addStream),e&&e.close(),{audioVideo:n,data:i,binaryBlob:r,binary:o,reliable:o,sctp:o,onnegotiationneeded:s}}(),validateId:function(e){return!e||/^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(e)},validateKey:function(e){return!e||/^[A-Za-z0-9]+(?:[ _-][A-Za-z0-9]+)*$/.exec(e)},debug:!1,inherits:function(e,t){e.super_=t,e.prototype=Object.create(t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}})},extend:function(e,t){for(var i in t)t.hasOwnProperty(i)&&(e[i]=t[i]);return e},pack:n.default.pack,unpack:n.default.unpack,log:function(){if(c.util.debug){var e=!1,t=Array.prototype.slice.call(arguments);t.unshift("PeerJS: ");for(var i=0,n=t.length;i<n;i++)t[i]instanceof Error&&(t[i]="("+t[i].name+") "+t[i].message,e=!0);e?console.error.apply(console,t):console.log.apply(console,t)}},setZeroTimeout:function(t){var i=[],n="zero-timeout-message";function e(e){e.source==t&&e.data==n&&(e.stopPropagation&&e.stopPropagation(),i.length&&i.shift()())}return t.addEventListener?t.addEventListener("message",e,!0):t.attachEvent&&t.attachEvent("onmessage",e),function(e){i.push(e),t.postMessage(n,"*")}}(window),chunk:function(e){for(var t,i=[],n=e.size,r=t=0,o=Math.ceil(n/c.util.chunkedMTU);r<n;){var s=Math.min(n,r+c.util.chunkedMTU),a=e.slice(r,s),u={__peerData:p,n:t,data:a,total:o};i.push(u),r=s,t+=1}return p+=1,i},blobToArrayBuffer:function(e,t){var i=new FileReader;i.onload=function(e){t(e.target.result)},i.readAsArrayBuffer(e)},blobToBinaryString:function(e,t){var i=new FileReader;i.onload=function(e){t(e.target.result)},i.readAsBinaryString(e)},binaryStringToArrayBuffer:function(e){for(var t=new Uint8Array(e.length),i=0;i<e.length;i++)t[i]=255&e.charCodeAt(i);return t.buffer},randomToken:function(){return Math.random().toString(36).substr(2)},isSecure:function(){return"https:"===location.protocol}}},{"./adapter":1,"js-binarypack":10}],9:[function(e,t,i){"use strict";function r(e,t,i){this.fn=e,this.context=t,this.once=i||!1}function n(){}n.prototype._events=void 0,n.prototype.listeners=function(e){if(!this._events||!this._events[e])return[];if(this._events[e].fn)return[this._events[e].fn];for(var t=0,i=this._events[e].length,n=new Array(i);t<i;t++)n[t]=this._events[e][t].fn;return n},n.prototype.emit=function(e,t,i,n,r,o){if(!this._events||!this._events[e])return!1;var s,a,u=this._events[e],c=arguments.length;if("function"==typeof u.fn){switch(u.once&&this.removeListener(e,u.fn,!0),c){case 1:return u.fn.call(u.context),!0;case 2:return u.fn.call(u.context,t),!0;case 3:return u.fn.call(u.context,t,i),!0;case 4:return u.fn.call(u.context,t,i,n),!0;case 5:return u.fn.call(u.context,t,i,n,r),!0;case 6:return u.fn.call(u.context,t,i,n,r,o),!0}for(a=1,s=new Array(c-1);a<c;a++)s[a-1]=arguments[a];u.fn.apply(u.context,s)}else{var p,l=u.length;for(a=0;a<l;a++)switch(u[a].once&&this.removeListener(e,u[a].fn,!0),c){case 1:u[a].fn.call(u[a].context);break;case 2:u[a].fn.call(u[a].context,t);break;case 3:u[a].fn.call(u[a].context,t,i);break;default:if(!s)for(p=1,s=new Array(c-1);p<c;p++)s[p-1]=arguments[p];u[a].fn.apply(u[a].context,s)}}return!0},n.prototype.on=function(e,t,i){var n=new r(t,i||this);return this._events||(this._events={}),this._events[e]?this._events[e].fn?this._events[e]=[this._events[e],n]:this._events[e].push(n):this._events[e]=n,this},n.prototype.once=function(e,t,i){var n=new r(t,i||this,!0);return this._events||(this._events={}),this._events[e]?this._events[e].fn?this._events[e]=[this._events[e],n]:this._events[e].push(n):this._events[e]=n,this},n.prototype.removeListener=function(e,t,i){if(!this._events||!this._events[e])return this;var n=this._events[e],r=[];if(t&&(n.fn&&(n.fn!==t||i&&!n.once)&&r.push(n),!n.fn))for(var o=0,s=n.length;o<s;o++)(n[o].fn!==t||i&&!n[o].once)&&r.push(n[o]);return r.length?this._events[e]=1===r.length?r[0]:r:delete this._events[e],this},n.prototype.removeAllListeners=function(e){return this._events&&(e?delete this._events[e]:this._events={}),this},n.prototype.off=n.prototype.removeListener,n.prototype.addListener=n.prototype.on,n.prototype.setMaxListeners=function(){return this},((n.EventEmitter=n).EventEmitter2=n).EventEmitter3=n,t.exports=n},{}],10:[function(e,t,i){var n=e("./bufferbuilder").BufferBuilder,r=e("./bufferbuilder").binaryFeatures,o={unpack:function(e){return new s(e).unpack()},pack:function(e){var t=new a;return t.pack(e),t.getBuffer()}};function s(e){this.index=0,this.dataBuffer=e,this.dataView=new Uint8Array(this.dataBuffer),this.length=this.dataBuffer.byteLength}function a(){this.bufferBuilder=new n}function u(e){var t=e.charCodeAt(0);return t<=2047?"00":t<=65535?"000":t<=2097151?"0000":t<=67108863?"00000":"000000"}t.exports=o,s.prototype.unpack=function(){var e,t=this.unpack_uint8();if(t<128)return t;if((224^t)<32)return(224^t)-32;if((e=160^t)<=15)return this.unpack_raw(e);if((e=176^t)<=15)return this.unpack_string(e);if((e=144^t)<=15)return this.unpack_array(e);if((e=128^t)<=15)return this.unpack_map(e);switch(t){case 192:return null;case 193:return;case 194:return!1;case 195:return!0;case 202:return this.unpack_float();case 203:return this.unpack_double();case 204:return this.unpack_uint8();case 205:return this.unpack_uint16();case 206:return this.unpack_uint32();case 207:return this.unpack_uint64();case 208:return this.unpack_int8();case 209:return this.unpack_int16();case 210:return this.unpack_int32();case 211:return this.unpack_int64();case 212:case 213:case 214:case 215:return;case 216:return e=this.unpack_uint16(),this.unpack_string(e);case 217:return e=this.unpack_uint32(),this.unpack_string(e);case 218:return e=this.unpack_uint16(),this.unpack_raw(e);case 219:return e=this.unpack_uint32(),this.unpack_raw(e);case 220:return e=this.unpack_uint16(),this.unpack_array(e);case 221:return e=this.unpack_uint32(),this.unpack_array(e);case 222:return e=this.unpack_uint16(),this.unpack_map(e);case 223:return e=this.unpack_uint32(),this.unpack_map(e)}},s.prototype.unpack_uint8=function(){var e=255&this.dataView[this.index];return this.index++,e},s.prototype.unpack_uint16=function(){var e=this.read(2),t=256*(255&e[0])+(255&e[1]);return this.index+=2,t},s.prototype.unpack_uint32=function(){var e=this.read(4),t=256*(256*(256*e[0]+e[1])+e[2])+e[3];return this.index+=4,t},s.prototype.unpack_uint64=function(){var e=this.read(8),t=256*(256*(256*(256*(256*(256*(256*e[0]+e[1])+e[2])+e[3])+e[4])+e[5])+e[6])+e[7];return this.index+=8,t},s.prototype.unpack_int8=function(){var e=this.unpack_uint8();return e<128?e:e-256},s.prototype.unpack_int16=function(){var e=this.unpack_uint16();return e<32768?e:e-65536},s.prototype.unpack_int32=function(){var e=this.unpack_uint32();return e<Math.pow(2,31)?e:e-Math.pow(2,32)},s.prototype.unpack_int64=function(){var e=this.unpack_uint64();return e<Math.pow(2,63)?e:e-Math.pow(2,64)},s.prototype.unpack_raw=function(e){if(this.length<this.index+e)throw new Error("BinaryPackFailure: index is out of range "+this.index+" "+e+" "+this.length);var t=this.dataBuffer.slice(this.index,this.index+e);return this.index+=e,t},s.prototype.unpack_string=function(e){for(var t,i,n=this.read(e),r=0,o="";r<e;)(t=n[r])<128?(o+=String.fromCharCode(t),r++):(192^t)<32?(i=(192^t)<<6|63&n[r+1],o+=String.fromCharCode(i),r+=2):(i=(15&t)<<12|(63&n[r+1])<<6|63&n[r+2],o+=String.fromCharCode(i),r+=3);return this.index+=e,o},s.prototype.unpack_array=function(e){for(var t=new Array(e),i=0;i<e;i++)t[i]=this.unpack();return t},s.prototype.unpack_map=function(e){for(var t={},i=0;i<e;i++){var n=this.unpack(),r=this.unpack();t[n]=r}return t},s.prototype.unpack_float=function(){var e=this.unpack_uint32(),t=(e>>23&255)-127;return(0==e>>31?1:-1)*(8388607&e|8388608)*Math.pow(2,t-23)},s.prototype.unpack_double=function(){var e=this.unpack_uint32(),t=this.unpack_uint32(),i=(e>>20&2047)-1023;return(0==e>>31?1:-1)*((1048575&e|1048576)*Math.pow(2,i-20)+t*Math.pow(2,i-52))},s.prototype.read=function(e){var t=this.index;if(t+e<=this.length)return this.dataView.subarray(t,t+e);throw new Error("BinaryPackFailure: read index out of range")},a.prototype.getBuffer=function(){return this.bufferBuilder.getBuffer()},a.prototype.pack=function(e){var t=typeof e;if("string"==t)this.pack_string(e);else if("number"==t)Math.floor(e)===e?this.pack_integer(e):this.pack_double(e);else if("boolean"==t)!0===e?this.bufferBuilder.append(195):!1===e&&this.bufferBuilder.append(194);else if("undefined"==t)this.bufferBuilder.append(192);else{if("object"!=t)throw new Error('Type "'+t+'" not yet supported');if(null===e)this.bufferBuilder.append(192);else{var i=e.constructor;if(i==Array)this.pack_array(e);else if(i==Blob||i==File)this.pack_bin(e);else if(i==ArrayBuffer)r.useArrayBufferView?this.pack_bin(new Uint8Array(e)):this.pack_bin(e);else if("BYTES_PER_ELEMENT"in e)r.useArrayBufferView?this.pack_bin(new Uint8Array(e.buffer)):this.pack_bin(e.buffer);else if(i==Object)this.pack_object(e);else if(i==Date)this.pack_string(e.toString());else{if("function"!=typeof e.toBinaryPack)throw new Error('Type "'+i.toString()+'" not yet supported');this.bufferBuilder.append(e.toBinaryPack())}}}this.bufferBuilder.flush()},a.prototype.pack_bin=function(e){var t=e.length||e.byteLength||e.size;if(t<=15)this.pack_uint8(160+t);else if(t<=65535)this.bufferBuilder.append(218),this.pack_uint16(t);else{if(!(t<=4294967295))throw new Error("Invalid length");this.bufferBuilder.append(219),this.pack_uint32(t)}this.bufferBuilder.append(e)},a.prototype.pack_string=function(e){var t,i=600<(t=e).length?new Blob([t]).size:t.replace(/[^\u0000-\u007F]/g,u).length;if(i<=15)this.pack_uint8(176+i);else if(i<=65535)this.bufferBuilder.append(216),this.pack_uint16(i);else{if(!(i<=4294967295))throw new Error("Invalid length");this.bufferBuilder.append(217),this.pack_uint32(i)}this.bufferBuilder.append(e)},a.prototype.pack_array=function(e){var t=e.length;if(t<=15)this.pack_uint8(144+t);else if(t<=65535)this.bufferBuilder.append(220),this.pack_uint16(t);else{if(!(t<=4294967295))throw new Error("Invalid length");this.bufferBuilder.append(221),this.pack_uint32(t)}for(var i=0;i<t;i++)this.pack(e[i])},a.prototype.pack_integer=function(e){if(-32<=e&&e<=127)this.bufferBuilder.append(255&e);else if(0<=e&&e<=255)this.bufferBuilder.append(204),this.pack_uint8(e);else if(-128<=e&&e<=127)this.bufferBuilder.append(208),this.pack_int8(e);else if(0<=e&&e<=65535)this.bufferBuilder.append(205),this.pack_uint16(e);else if(-32768<=e&&e<=32767)this.bufferBuilder.append(209),this.pack_int16(e);else if(0<=e&&e<=4294967295)this.bufferBuilder.append(206),this.pack_uint32(e);else if(-2147483648<=e&&e<=2147483647)this.bufferBuilder.append(210),this.pack_int32(e);else if(-0x8000000000000000<=e&&e<=0x8000000000000000)this.bufferBuilder.append(211),this.pack_int64(e);else{if(!(0<=e&&e<=0x10000000000000000))throw new Error("Invalid integer");this.bufferBuilder.append(207),this.pack_uint64(e)}},a.prototype.pack_double=function(e){var t=0;e<0&&(t=1,e=-e);var i=Math.floor(Math.log(e)/Math.LN2),n=e/Math.pow(2,i)-1,r=Math.floor(n*Math.pow(2,52)),o=Math.pow(2,32),s=t<<31|i+1023<<20|r/o&1048575,a=r%o;this.bufferBuilder.append(203),this.pack_int32(s),this.pack_int32(a)},a.prototype.pack_object=function(e){var t=Object.keys(e).length;if(t<=15)this.pack_uint8(128+t);else if(t<=65535)this.bufferBuilder.append(222),this.pack_uint16(t);else{if(!(t<=4294967295))throw new Error("Invalid length");this.bufferBuilder.append(223),this.pack_uint32(t)}for(var i in e)e.hasOwnProperty(i)&&(this.pack(i),this.pack(e[i]))},a.prototype.pack_uint8=function(e){this.bufferBuilder.append(e)},a.prototype.pack_uint16=function(e){this.bufferBuilder.append(e>>8),this.bufferBuilder.append(255&e)},a.prototype.pack_uint32=function(e){var t=4294967295&e;this.bufferBuilder.append((4278190080&t)>>>24),this.bufferBuilder.append((16711680&t)>>>16),this.bufferBuilder.append((65280&t)>>>8),this.bufferBuilder.append(255&t)},a.prototype.pack_uint64=function(e){var t=e/Math.pow(2,32),i=e%Math.pow(2,32);this.bufferBuilder.append((4278190080&t)>>>24),this.bufferBuilder.append((16711680&t)>>>16),this.bufferBuilder.append((65280&t)>>>8),this.bufferBuilder.append(255&t),this.bufferBuilder.append((4278190080&i)>>>24),this.bufferBuilder.append((16711680&i)>>>16),this.bufferBuilder.append((65280&i)>>>8),this.bufferBuilder.append(255&i)},a.prototype.pack_int8=function(e){this.bufferBuilder.append(255&e)},a.prototype.pack_int16=function(e){this.bufferBuilder.append((65280&e)>>8),this.bufferBuilder.append(255&e)},a.prototype.pack_int32=function(e){this.bufferBuilder.append(e>>>24&255),this.bufferBuilder.append((16711680&e)>>>16),this.bufferBuilder.append((65280&e)>>>8),this.bufferBuilder.append(255&e)},a.prototype.pack_int64=function(e){var t=Math.floor(e/Math.pow(2,32)),i=e%Math.pow(2,32);this.bufferBuilder.append((4278190080&t)>>>24),this.bufferBuilder.append((16711680&t)>>>16),this.bufferBuilder.append((65280&t)>>>8),this.bufferBuilder.append(255&t),this.bufferBuilder.append((4278190080&i)>>>24),this.bufferBuilder.append((16711680&i)>>>16),this.bufferBuilder.append((65280&i)>>>8),this.bufferBuilder.append(255&i)}},{"./bufferbuilder":11}],11:[function(e,t,i){var n={};n.useBlobBuilder=function(){try{return new Blob([]),!1}catch(e){return!0}}(),n.useArrayBufferView=!n.useBlobBuilder&&function(){try{return 0===new Blob([new Uint8Array([])]).size}catch(e){return!0}}(),t.exports.binaryFeatures=n;var r=t.exports.BlobBuilder;function o(){this._pieces=[],this._parts=[]}"undefined"!=typeof window&&(r=t.exports.BlobBuilder=window.WebKitBlobBuilder||window.MozBlobBuilder||window.MSBlobBuilder||window.BlobBuilder),o.prototype.append=function(e){"number"==typeof e?this._pieces.push(e):(this.flush(),this._parts.push(e))},o.prototype.flush=function(){if(0<this._pieces.length){var e=new Uint8Array(this._pieces);n.useArrayBufferView||(e=e.buffer),this._parts.push(e),this._pieces=[]}},o.prototype.getBuffer=function(){if(this.flush(),n.useBlobBuilder){for(var e=new r,t=0,i=this._parts.length;t<i;t++)e.append(this._parts[t]);return e.getBlob()}return new Blob(this._parts)},t.exports.BufferBuilder=o},{}],12:[function(e,t,i){var c=e("./util");function n(e,t){if(!(this instanceof n))return new n(e);this._dc=e,c.debug=t,this._outgoing={},this._incoming={},this._received={},this._window=1e3,this._mtu=500,this._interval=0,this._count=0,this._queue=[],this._setupDC()}n.prototype.send=function(e){var t=c.pack(e);t.size<this._mtu?this._handleSend(["no",t]):(this._outgoing[this._count]={ack:0,chunks:this._chunk(t)},c.debug&&(this._outgoing[this._count].timer=new Date),this._sendWindowedChunks(this._count),this._count+=1)},n.prototype._setupInterval=function(){var n=this;this._timeout=setInterval(function(){var e=n._queue.shift();if(e._multiple)for(var t=0,i=e.length;t<i;t+=1)n._intervalSend(e[t]);else n._intervalSend(e)},this._interval)},n.prototype._intervalSend=function(e){var t=this;e=c.pack(e),c.blobToBinaryString(e,function(e){t._dc.send(e)}),0===t._queue.length&&(clearTimeout(t._timeout),t._timeout=null)},n.prototype._processAcks=function(){for(var e in this._outgoing)this._outgoing.hasOwnProperty(e)&&this._sendWindowedChunks(e)},n.prototype._handleSend=function(e){for(var t=!0,i=0,n=this._queue.length;i<n;i+=1){var r=this._queue[i];r===e?t=!1:r._multiple&&-1!==r.indexOf(e)&&(t=!1)}t&&(this._queue.push(e),this._timeout||this._setupInterval())},n.prototype._setupDC=function(){var n=this;this._dc.onmessage=function(e){var t=e.data;if(t.constructor===String){var i=c.binaryStringToArrayBuffer(t);t=c.unpack(i),n._handleMessage(t)}}},n.prototype._handleMessage=function(e){var t,i=e[1],n=this._incoming[i],r=this._outgoing[i];switch(e[0]){case"no":var o=i;o&&this.onmessage(c.unpack(o));break;case"end":if(t=n,this._received[i]=e[2],!t)break;this._ack(i);break;case"ack":if(t=r){var s=e[2];t.ack=Math.max(s,t.ack),t.ack>=t.chunks.length?(c.log("Time: ",new Date-t.timer),delete this._outgoing[i]):this._processAcks()}break;case"chunk":if(!(t=n)){if(!0===this._received[i])break;t={ack:["ack",i,0],chunks:[]},this._incoming[i]=t}var a=e[2],u=e[3];t.chunks[a]=new Uint8Array(u),a===t.ack[2]&&this._calculateNextAck(i),this._ack(i);break;default:this._handleSend(e)}},n.prototype._chunk=function(e){for(var t=[],i=e.size,n=0;n<i;){var r=Math.min(i,n+this._mtu),o={payload:e.slice(n,r)};t.push(o),n=r}return c.log("Created",t.length,"chunks."),t},n.prototype._ack=function(e){var t=this._incoming[e].ack;this._received[e]===t[2]&&(this._complete(e),this._received[e]=!0),this._handleSend(t)},n.prototype._calculateNextAck=function(e){for(var t=this._incoming[e],i=t.chunks,n=0,r=i.length;n<r;n+=1)if(void 0===i[n])return void(t.ack[2]=n);t.ack[2]=i.length},n.prototype._sendWindowedChunks=function(e){c.log("sendWindowedChunks for: ",e);for(var t=this._outgoing[e],i=t.chunks,n=[],r=Math.min(t.ack+this._window,i.length),o=t.ack;o<r;o+=1)i[o].sent&&o!==t.ack||(i[o].sent=!0,n.push(["chunk",e,o,i[o].payload]));t.ack+this._window>=i.length&&n.push(["end",e,i.length]),n._multiple=!0,this._handleSend(n)},n.prototype._complete=function(e){c.log("Completed called for",e);var t=this,i=this._incoming[e].chunks,n=new Blob(i);c.blobToArrayBuffer(n,function(e){t.onmessage(c.unpack(e))}),delete this._incoming[e]},n.higherBandwidthSDP=function(e){var t=navigator.appVersion.match(/Chrome\/(.*?) /);if(t&&(t=parseInt(t[1].split(".").shift()))<31){var i=e.split("b=AS:30");if(1<i.length)return i[0]+"b=AS:102400"+i[1]}return e},n.prototype.onmessage=function(e){},t.exports.Reliable=n},{"./util":13}],13:[function(e,t,i){var n=e("js-binarypack"),r={debug:!1,inherits:function(e,t){e.super_=t,e.prototype=Object.create(t.prototype,{constructor:{value:e,enumerable:!1,writable:!0,configurable:!0}})},extend:function(e,t){for(var i in t)t.hasOwnProperty(i)&&(e[i]=t[i]);return e},pack:n.pack,unpack:n.unpack,log:function(){if(r.debug){for(var e=[],t=0;t<arguments.length;t++)e[t]=arguments[t];e.unshift("Reliable: "),console.log.apply(console,e)}},setZeroTimeout:function(t){var i=[],n="zero-timeout-message";function e(e){e.source==t&&e.data==n&&(e.stopPropagation&&e.stopPropagation(),i.length&&i.shift()())}return t.addEventListener?t.addEventListener("message",e,!0):t.attachEvent&&t.attachEvent("onmessage",e),function(e){i.push(e),t.postMessage(n,"*")}}(this),blobToArrayBuffer:function(e,t){var i=new FileReader;i.onload=function(e){t(e.target.result)},i.readAsArrayBuffer(e)},blobToBinaryString:function(e,t){var i=new FileReader;i.onload=function(e){t(e.target.result)},i.readAsBinaryString(e)},binaryStringToArrayBuffer:function(e){for(var t=new Uint8Array(e.length),i=0;i<e.length;i++)t[i]=255&e.charCodeAt(i);return t.buffer},randomToken:function(){return Math.random().toString(36).substr(2)}};t.exports=r},{"js-binarypack":10}]},{},[3]); \ No newline at end of file diff --git a/lib/spectrum.js b/lib/spectrum.js index adb2694c..ca1ed4e9 100644 --- a/lib/spectrum.js +++ b/lib/spectrum.js @@ -500,6 +500,7 @@ } function addColorToSelectionPalette(color) { + if (showSelectionPalette) { var rgb = tinycolor(color).toRgbString(); if (!paletteLookup[rgb] && $.inArray(rgb, selectionPalette) === -1) { @@ -559,7 +560,7 @@ } function dragStart() { - if (dragHeight <= 0 || dragWidth <= 0 || slideHeight <= 0) { + if (dragHeight <= 0 || dragWidth <= 0 || slideHeight <= 0 || flat) { reflow(); } isDragging = true; diff --git a/main.js b/main.js index 23c169b9..65c4fbe1 100644 --- a/main.js +++ b/main.js @@ -4,7 +4,7 @@ const url = require('url') let orig_win; -function createWindow() { +function createWindow(second_instance) { if (app.requestSingleInstanceLock && !app.requestSingleInstanceLock()) { return; } @@ -66,12 +66,15 @@ function createWindow() { win.on('closed', () => { win = null }) - //win.webContents.openDevTools() + if (second_instance === true) { + win.webContents.second_instance = true + + } } app.on('second-instance', function (event, argv, cwd) { process.argv = argv - createWindow() + createWindow(true) }) app.commandLine.appendSwitch('ignore-gpu-blacklist') diff --git a/package.json b/package.json index 26ce43f2..4857beea 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "Blockbench", "description": "Minecraft Block Model Editor", - "version": "2.5.1", + "version": "2.6.0", "license": "MIT", "author": { "name": "JannisX11",