2022-05-19 05:44:00 +08:00
const StartScreen = {
loaders : { }
} ;
2021-11-14 06:19:23 +08:00
function addStartScreenSection ( id , data ) {
if ( typeof id == 'object' ) {
data = id ;
id = '' ;
}
2022-03-16 23:19:56 +08:00
var obj = $ ( Interface . createElement ( 'section' , { id } ) )
2021-11-14 06:19:23 +08:00
if ( typeof data . graphic === 'object' ) {
var left = $ ( '<div class="start_screen_left graphic"></div>' )
obj . append ( left )
if ( data . graphic . type === 'icon' ) {
var icon = Blockbench . getIconNode ( data . graphic . icon )
left . addClass ( 'graphic_icon' )
left . append ( icon )
} else {
left . css ( 'background-image' , ` url(' ${ data . graphic . source } ') ` )
}
if ( data . graphic . width ) {
left . css ( 'width' , data . graphic . width + 'px' ) ;
}
if ( data . graphic . width && data . text ) {
left . css ( 'flex-shrink' , '0' ) ;
}
if ( data . graphic . width && data . graphic . height && Blockbench . isMobile ) {
left . css ( 'height' , '0' )
. css ( 'padding-top' , '0' )
. css ( 'padding-bottom' , ( data . graphic . height / data . graphic . width * 100 ) + '%' )
} else {
if ( data . graphic . height ) left . css ( 'height' , data . graphic . height + 'px' ) ;
if ( data . graphic . width && ! data . graphic . height && ! data . graphic . aspect _ratio ) left . css ( 'height' , data . graphic . width + 'px' ) ;
if ( data . graphic . aspect _ratio ) left . css ( 'aspect-ratio' , data . graphic . aspect _ratio ) ;
}
if ( data . graphic . description ) {
let content = $ ( marked ( data . graphic . description ) ) ;
content . css ( {
'bottom' : '15px' ,
'right' : '15px' ,
2021-12-21 22:30:01 +08:00
'color' : data . graphic . text _color || '#ffffff' ,
2021-11-14 06:19:23 +08:00
} ) ;
left . append ( content ) ;
}
}
if ( data . text instanceof Array ) {
var right = $ ( '<div class="start_screen_right"></div>' )
obj . append ( right )
data . text . forEach ( line => {
var content = line . text ? marked ( tl ( line . text ) ) : '' ;
switch ( line . type ) {
case 'h1' : var tag = 'h2' ; break ;
case 'h2' : var tag = 'h3' ; break ;
case 'list' :
var tag = 'ul class="list_style"' ;
line . list . forEach ( string => {
content += ` <li> ${ marked ( tl ( string ) ) } </li> ` ;
} )
break ;
case 'button' : var tag = 'button' ; break ;
default : var tag = 'p' ; break ;
}
var l = $ ( ` < ${ tag } > ${ content } </ ${ tag . split ( ' ' ) [ 0 ] } > ` ) ;
if ( typeof line . click == 'function' ) {
l . on ( 'click' , line . click ) ;
}
right . append ( l ) ;
} )
}
if ( data . layout == 'vertical' ) {
obj . addClass ( 'vertical' ) ;
}
if ( data . features instanceof Array ) {
let features _section = document . createElement ( 'ul' ) ;
features _section . className = 'start_screen_features'
data . features . forEach ( feature => {
let li = document . createElement ( 'li' ) ;
let img = new Image ( ) ; img . src = feature . image ;
let title = document . createElement ( 'h3' ) ; title . textContent = feature . title ;
let text = document . createElement ( 'p' ) ; text . textContent = feature . text ;
li . append ( img , title , text ) ;
features _section . append ( li ) ;
} )
obj . append ( features _section ) ;
}
if ( data . closable !== false ) {
obj . append ( ` <i class="material-icons start_screen_close_button">clear</i> ` ) ;
obj . find ( 'i.start_screen_close_button' ) . click ( ( e ) => {
obj . detach ( )
} ) ;
}
if ( typeof data . click == 'function' ) {
obj . on ( 'click' , event => {
if ( event . target . classList . contains ( 'start_screen_close_button' ) ) return ;
data . click ( )
} )
}
if ( data . color ) {
obj . css ( 'background-color' , data . color ) ;
if ( data . color == 'var(--color-bright_ui)' ) {
obj . addClass ( 'bright_ui' )
}
}
if ( data . text _color ) {
obj . css ( 'color' , data . text _color ) ;
}
if ( data . last ) {
$ ( '#start_screen content' ) . append ( obj ) ;
2022-03-28 19:04:23 +08:00
} else if ( data . insert _after ) {
$ ( '#start_screen content' ) . find ( ` # ${ data . insert _after } ` ) . after ( obj ) ;
} else if ( data . insert _before ) {
$ ( '#start_screen content' ) . find ( ` # ${ data . insert _before } ` ) . before ( obj ) ;
2021-11-14 06:19:23 +08:00
} else {
$ ( '#start_screen content' ) . prepend ( obj ) ;
}
2022-03-28 19:04:23 +08:00
if ( ! obj [ 0 ] . parentElement ) {
$ ( '#start_screen content' ) . append ( obj ) ;
}
2022-04-05 04:56:51 +08:00
return {
delete ( ) {
obj [ 0 ] . remove ( ) ;
}
}
2021-11-14 06:19:23 +08:00
}
onVueSetup ( function ( ) {
StateMemory . init ( 'start_screen_list_type' , 'string' )
StartScreen . vue = new Vue ( {
el : '#start_screen' ,
data : {
formats : Formats ,
2022-05-19 05:44:00 +08:00
loaders : StartScreen . loaders ,
selected _format : '' ,
2021-11-14 06:19:23 +08:00
recent : isApp ? recent _projects : [ ] ,
list _type : StateMemory . start _screen _list _type || 'grid' ,
redact _names : settings . streamer _mode . value ,
redacted : tl ( 'generic.redacted' ) ,
search _term : '' ,
isApp ,
2022-05-19 05:44:00 +08:00
mobile _layout : Blockbench . isMobile ,
2021-11-14 06:19:23 +08:00
getIconNode : Blockbench . getIconNode
} ,
methods : {
getDate ( p ) {
if ( p . day ) {
var diff = ( 365e10 + Blockbench . openTime . dayOfYear ( ) - p . day ) % 365 ;
if ( diff <= 0 ) {
return tl ( 'dates.today' ) ;
} else if ( diff == 1 ) {
return tl ( 'dates.yesterday' ) ;
} else if ( diff <= 7 ) {
return tl ( 'dates.this_week' ) ;
} else {
return tl ( 'dates.weeks_ago' , [ Math . ceil ( diff / 7 ) ] ) ;
}
} else {
return '-'
}
} ,
openProject : function ( p , event ) {
Blockbench . read ( [ p . path ] , { } , files => {
loadModelFile ( files [ 0 ] ) ;
} )
} ,
getThumbnail ( model _path ) {
let hash = model _path . hashCode ( ) . toString ( ) . replace ( /^-/ , '0' ) ;
let path = PathModule . join ( app . getPath ( 'userData' ) , 'thumbnails' , ` ${ hash } .png ` ) ;
if ( ! fs . existsSync ( path ) ) return ;
return path + '?' + Math . round ( Math . random ( ) * 255 ) ;
} ,
setListType ( type ) {
this . list _type = type ;
StateMemory . start _screen _list _type = type ;
StateMemory . save ( 'start_screen_list_type' )
} ,
2022-05-19 05:44:00 +08:00
getFormatCategories ( ) {
let categories = { } ;
function add ( key , format ) {
if ( ! categories [ format . category ] ) {
categories [ format . category ] = {
name : tl ( 'format_category.' + format . category ) ,
entries : [ ]
}
}
categories [ format . category ] . entries . push ( format ) ;
}
for ( let key in this . formats ) {
add ( key , this . formats [ key ] )
}
for ( let key in this . loaders ) {
add ( key , this . loaders [ key ] )
}
return categories ;
} ,
loadFormat ( format _entry ) {
this . selected _format = format _entry . id ;
if ( format _entry . onSetup ) format _entry . onSetup ( ) ;
} ,
confirmSetupScreen ( format _entry ) {
this . selected _format = '' ;
if ( format _entry . onStart ) format _entry . onStart ( ) ;
if ( typeof format _entry . new == 'function' ) format _entry . new ( ) ;
} ,
2021-11-14 06:19:23 +08:00
tl
} ,
computed : {
projects ( ) {
if ( ! this . search _term ) return this . recent ;
let terms = this . search _term . toLowerCase ( ) . split ( /\s/ ) ;
2021-11-15 06:47:56 +08:00
2021-11-14 06:19:23 +08:00
return this . recent . filter ( project => {
return ! terms . find ( term => (
! project . path . toLowerCase ( ) . includes ( term )
) )
} )
}
} ,
template : `
< div id = "start_screen" >
< content >
2022-05-19 05:44:00 +08:00
< section id = "start_files" >
< div class = "start_screen_left" v - if = "!(selected_format && mobile_layout)" >
2021-11-14 06:19:23 +08:00
< h2 > $ { tl ( 'mode.start.new' ) } < / h 2 >
< div class = "bar next_to_title" >
< div class = "tool" onclick = "Blockbench.openLink('https://blockbench.net/quickstart/')" >
< div class = "tooltip" > $ { tl ( 'menu.help.quickstart' ) } < / d i v >
< i class = "fas fa-question-circle" > < / i >
< / d i v >
< / d i v >
< ul >
2022-05-19 05:44:00 +08:00
< li v - for = "(category, key) in getFormatCategories()" class = "format_category" : key = "key" >
< label > { { category . name } } < / l a b e l >
< ul >
< li
v - for = "format_entry in category.entries" : key = "format_entry.id"
class = "format_entry" : class = "{[format_entry instanceof ModelFormat ? 'format' : 'loader']: true, selected: format_entry.id == selected_format}"
: title = "format_entry.description"
v - if = "format_entry.show_on_start_screen && (!redact_names || !format_entry.confidential)"
@ click = "loadFormat(format_entry)"
@ dblclick = "confirmSetupScreen(format_entry)"
>
< span class = "icon_wrapper f_left" v - html = "getIconNode(format_entry.icon).outerHTML" > < / s p a n >
< label > { { format _entry . name } } < / l a b e l >
< / l i >
< / u l >
2021-11-14 06:19:23 +08:00
< / l i >
< / u l >
< / d i v >
2022-05-19 05:44:00 +08:00
< div class = "start_screen_right start_screen_format_setup" v - if = "viewed_format = (selected_format && (formats[selected_format] || loaders[selected_format]) )" >
< div class = "tool format_setup_close_button" @ click = "selected_format = ''" > < i class = "material-icons" > arrow _back _ios < / i > < / d i v >
< h2 style = "margin-bottom: 12px;" > { { viewed _format . name } } < / h 2 >
< p class = "format_description" v - if = "viewed_format.description" > { { viewed _format . description } } < / p >
< div v - if = "viewed_format.about" class = "about format_about" v - html = "marked(viewed_format.about.replace(/\\n/g, '\\n\\n'))" > < button > < / b u t t o n > < / d i v >
< img src = "assets/start_screen/generic.png" height = "225px" / >
< p class = "format_target" v - if = "viewed_format.target" >
< b > $ { tl ( 'mode.start.target' ) } < / b > :
< template v - if = "viewed_format.target instanceof Array" >
< span v - for = "target in viewed_format.target" > { { target } } < / s p a n >
< / t e m p l a t e >
< span v - else > { { viewed _format . target } } < / s p a n >
< / p >
< div class = "button_bar" >
< button style = "margin-top: 20px;" @ click = "confirmSetupScreen(viewed_format)" > $ { tl ( 'mode.start.start' ) } < / b u t t o n >
< / d i v >
< / d i v >
< div class = "start_screen_right" v - else >
2021-11-14 06:19:23 +08:00
< h2 class = "tl" > $ { tl ( 'mode.start.recent' ) } < / h 2 >
< div id = "start_screen_view_menu" v - if = "isApp && !redact_names" >
< search - bar : hide = "true" v - model = "search_term" > < / s e a r c h - b a r >
< li class = "tool" v - bind : class = "{selected: list_type == 'grid'}" v - on : click = "setListType('grid')" >
< i class = "material-icons" > view _module < / i >
< / l i >
< li class = "tool" v - bind : class = "{selected: list_type == 'list'}" v - on : click = "setListType('list')" >
< i class = "material-icons" > list < / i >
< / l i >
< / d i v >
< div v - if = "redact_names" > { { '[' + tl ( 'generic.redacted' ) + ']' } } < / d i v >
< ul v - else - if = "list_type == 'list'" >
< li v - on : click = "openProject(project, $event)" v - for = "project in projects" : key = "project.path" v - bind : title = "redact_names ? '' : project.path" class = "recent_project" >
< span class = "icon_wrapper" v - html = "getIconNode(project.icon).outerHTML" > < / s p a n >
< span class = "recent_project_name" > { { redact _names ? redacted : project . name } } < / s p a n >
< span class = "recent_project_date" > { { getDate ( project ) } } < / s p a n >
< / l i >
< div v - if = "recent.length == 0" > { { tl ( 'mode.start.no_recents' ) } } < / d i v >
< / u l >
< ul : class = "{redact: redact_names}" style = "display: grid;" v - else >
2021-11-15 06:47:56 +08:00
< li v - on : click = "openProject(project, $event)" v - for = "project in projects" : key = "project.path" v - bind : title = "redact_names ? '' : project.path" class = "recent_project thumbnail" >
< img class = "thumbnail_image" v - if = "getThumbnail(project.path)" : src = "getThumbnail(project.path)" / >
< span class = "recent_project_name" > { { redact _names ? redacted : project . name } } < / s p a n >
< span class = "icon_wrapper" v - html = "getIconNode(project.icon).outerHTML" > < / s p a n >
< / l i >
2021-11-14 06:19:23 +08:00
< / u l >
2022-05-19 05:44:00 +08:00
< div class = "button_bar" >
< button style = "margin-top: 20px;" onclick = "BarItems.open_model.trigger()" > $ { tl ( 'action.open_model' ) } < / b u t t o n >
< / d i v >
2021-11-14 06:19:23 +08:00
< / d i v >
2022-05-19 05:44:00 +08:00
2021-11-14 06:19:23 +08:00
< / s e c t i o n >
< / c o n t e n t >
< / d i v >
`
} )
} ) ;
2022-05-19 05:44:00 +08:00
StartScreen . addLoader = function ( id , options ) {
let loader = {
id ,
name : tl ( options . name ) ,
description : options . description ? tl ( options . description ) : '' ,
icon : options . icon || 'arrow_forward' ,
category : options . category || 'loaders' ,
target : options . category || 'loaders' ,
show _on _start _screen : true ,
confidential : options . confidential || false ,
condition : options . condition ,
onSetup : options . onSetup ,
onStart : options . onStart ,
} ;
Vue . set ( StartScreen . vue . loaders , id , loader ) ;
} ;
2021-11-14 06:19:23 +08:00
( function ( ) {
/ * $ . g e t J S O N ( ' . / c o n t e n t / n e w s . j s o n ' ) . t h e n ( d a t a = > {
addStartScreenSection ( 'new_version' , data . new _version )
} ) * /
2022-01-08 21:22:22 +08:00
var news _call = $ . ajax ( {
cache : false ,
url : 'https://web.blockbench.net/content/news.json' ,
dataType : 'json'
} ) ;
2021-11-14 06:19:23 +08:00
documentReady . then ( ( ) => {
Blockbench . startup _count = parseInt ( localStorage . getItem ( 'startups' ) || 0 )
//Backup Model
if ( localStorage . getItem ( 'backup_model' ) && ( ! isApp || ! currentwindow . webContents . second _instance ) ) {
var backup _model = localStorage . getItem ( 'backup_model' )
localStorage . removeItem ( 'backup_model' )
addStartScreenSection ( {
color : 'var(--color-back)' ,
graphic : { type : 'icon' , icon : 'fa-archive' } ,
text : [
{ type : 'h2' , text : tl ( 'message.recover_backup.title' ) } ,
{ text : tl ( 'message.recover_backup.message' ) } ,
{ type : 'button' , text : tl ( 'dialog.ok' ) , click : ( e ) => {
loadModelFile ( { content : backup _model , path : 'backup.bbmodel' , no _file : true } )
} }
]
} )
}
if ( settings . streamer _mode . value ) {
updateStreamerModeNotification ( )
}
2022-03-28 19:04:23 +08:00
addStartScreenSection ( 'splash_screen' , {
"text_color" : '#000000' ,
"graphic" : {
"type" : "image" ,
"source" : "./assets/splash_art.png?42" ,
"width" : 1000 ,
"aspect_ratio" : "21/9" ,
"description" : "Splash Art by [Shroomy](https://twitter.com/ShroomyArts) and [RedstoneMvv](https://twitter.com/Redstone_mvv)" ,
"text_color" : '#cfcfcf'
}
} )
if ( ! Blockbench . hasFlag ( 'after_update' ) ) {
document . getElementById ( 'start_screen' ) . scrollTop = 100 ;
}
2021-11-14 06:19:23 +08:00
//Twitter
let twitter _ad ;
if ( Blockbench . startup _count < 20 && Blockbench . startup _count % 5 === 4 ) {
twitter _ad = true ;
addStartScreenSection ( {
color : '#1da1f2' ,
text _color : '#ffffff' ,
graphic : { type : 'icon' , icon : 'fab.fa-twitter' } ,
text : [
{ type : 'h2' , text : 'Blockbench on Twitter' } ,
{ text : 'Follow Blockbench on Twitter for the latest news as well as cool models from the community! [twitter.com/blockbench](https://twitter.com/blockbench/)' }
] ,
last : true
} )
}
//Discord
if ( Blockbench . startup _count < 6 && ! twitter _ad ) {
addStartScreenSection ( {
color : '#5865F2' ,
text _color : '#ffffff' ,
graphic : { type : 'icon' , icon : 'fab.fa-discord' } ,
text : [
{ type : 'h2' , text : 'Discord Server' } ,
{ text : 'You need help with modeling or you want to chat about Blockbench? Join the official [Blockbench Discord](https://discord.gg/WVHg5kH)!' }
] ,
last : true
} )
}
2022-03-06 05:59:54 +08:00
// Quick Setup
2022-03-06 19:22:59 +08:00
if ( Blockbench . startup _count <= 1 ) {
2021-11-14 06:19:23 +08:00
2022-03-06 05:59:54 +08:00
let section = Interface . createElement ( 'section' , { id : 'quick_setup' } ) ;
$ ( '#start_screen content' ) . prepend ( section ) ;
2021-11-14 06:19:23 +08:00
2022-03-06 05:59:54 +08:00
new Vue ( {
data ( ) { return {
language : Language . code ,
language _original : Language . code ,
languages : Language . options ,
keymap : 'default' ,
keymap _changed : false ,
theme : 'dark' ,
2022-05-15 02:37:35 +08:00
keymap _options : {
default : tl ( 'action.load_keymap.default' ) ,
mouse : tl ( 'action.load_keymap.mouse' ) ,
blender : 'Blender' ,
cinema4d : 'Cinema 4D' ,
maya : 'Maya' ,
} ,
2022-03-06 05:59:54 +08:00
} } ,
methods : {
tl ,
close ( ) {
obj . remove ( ) ;
} ,
reload ( ) {
Blockbench . reload ( ) ;
} ,
loadTheme ( theme _id ) {
this . theme = theme _id ;
let theme = CustomTheme . themes . find ( t => t . id == theme _id ) ;
if ( theme ) CustomTheme . loadTheme ( theme ) ;
} ,
getThemeThumbnailStyle ( theme _id ) {
let theme = CustomTheme . themes . find ( t => t . id == theme _id ) ;
let style = { } ;
if ( ! theme ) return style ;
for ( let key in theme . colors ) {
style [ ` --color- ${ key } ` ] = theme . colors [ key ] ;
}
return style ;
} ,
openThemes ( ) {
BarItems . theme _window . click ( ) ;
}
} ,
watch : {
language ( v ) {
settings . language . set ( v ) ;
Settings . save ( ) ;
} ,
keymap ( keymap , old _keymap ) {
this . keymap _changed = true ;
let success = Keybinds . loadKeymap ( keymap , true ) ;
if ( ! success ) this . keymap = old _keymap ;
}
} ,
template : `
< section id = "quick_setup" >
< i class = "material-icons start_screen_close_button" @ click = "close()" > clear < / i >
< h2 > $ { tl ( 'mode.start.quick_setup' ) } < / h 2 >
2021-11-14 06:19:23 +08:00
2022-03-06 05:59:54 +08:00
< div >
< label > $ { tl ( 'mode.start.keymap' ) } : < / l a b e l >
2022-05-15 02:37:35 +08:00
< select - input v - model = "keymap" : options = "keymap_options" / >
2022-03-06 05:59:54 +08:00
< p v - if = "keymap_changed" > { { tl ( 'action.load_keymap.' + keymap + '.desc' ) } } < / p >
< / d i v >
< div >
< label > $ { tl ( 'settings.language' ) } : < / l a b e l >
2022-05-15 02:37:35 +08:00
< select - input v - model = "language" : options = "languages" / >
2022-03-06 05:59:54 +08:00
< div class = "tool" @ click = "reload()" v - if = "language != language_original" : title = "tl('action.reload')" >
< i class = "material-icons" > refresh < / i >
< / d i v >
< p v - if = "language != language_original" > { { tl ( 'message.restart_to_update' ) } } < / p >
< / d i v >
< div style = "width: 640px;" >
< label > $ { tl ( 'dialog.settings.theme' ) } : < / l a b e l >
< div class = "quick_setup_theme" : class = "{selected: theme == 'dark'}" @ click = "loadTheme('dark')" > < div : style = "getThemeThumbnailStyle('dark')" > < / d i v > D a r k < / d i v >
< div class = "quick_setup_theme" : class = "{selected: theme == 'light'}" @ click = "loadTheme('light')" > < div : style = "getThemeThumbnailStyle('light')" > < / d i v > L i g h t < / d i v >
< div class = "quick_setup_theme" : class = "{selected: theme == 'contrast'}" @ click = "loadTheme('contrast')" > < div : style = "getThemeThumbnailStyle('contrast')" > < / d i v > C o n t r a s t < / d i v >
< div class = "quick_setup_theme more_themes" @ click = "openThemes()" > < div > < i class = "material-icons" > more _horiz < / i > < / d i v > { { t l ( ' m o d e . s t a r t . q u i c k _ s e t u p . m o r e _ t h e m e s ' ) } } < / d i v >
< / d i v >
< / s e c t i o n >
`
} ) . $mount ( section ) ;
2021-11-14 06:19:23 +08:00
}
} )
2022-03-28 19:04:23 +08:00
Promise . all ( [ news _call , documentReady ] ) . then ( ( data ) => {
if ( ! data || ! data [ 0 ] ) return ;
data = data [ 0 ] ;
//Update Screen
if ( Blockbench . hasFlag ( 'after_update' ) && data . new _version ) {
data . new _version . insert _after = 'splash_screen'
addStartScreenSection ( 'new_version' , data . new _version ) ;
jQuery . ajax ( {
url : 'https://blckbn.ch/api/event/successful_update' ,
type : 'POST' ,
data : {
version : Blockbench . version
}
} )
}
if ( data . psa ) {
( function ( ) {
if ( typeof data . psa . version == 'string' ) {
if ( data . psa . version . includes ( '-' ) ) {
limits = data . psa . version . split ( '-' ) ;
if ( limits [ 0 ] && compareVersions ( limits [ 0 ] , Blockbench . version ) ) return ;
if ( limits [ 1 ] && compareVersions ( Blockbench . version , limits [ 1 ] ) ) return ;
} else {
if ( data . psa . version != Blockbench . version ) return ;
}
}
addStartScreenSection ( data . psa )
} ) ( )
}
} )
2021-11-14 06:19:23 +08:00
} ) ( )