123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094 |
- namespace eval osd_menu {
- set_help_text main_menu_open "Show the OSD menu."
- set_help_text main_menu_close "Remove the OSD menu."
- set_help_text main_menu_toggle "Toggle the OSD menu."
- variable is_dingux [string match dingux "[openmsx_info platform]"]
- variable scaling_available [expr {[lindex [lindex [openmsx_info setting scale_factor] 2] 1] > 1}]
- proc get_optional {dict_name key default} {
- upvar $dict_name d
- expr {[dict exists $d $key] ? [dict get $d $key] : $default}
- }
- proc set_optional {dict_name key value} {
- upvar $dict_name d
- if {![dict exists $d $key]} {
- dict set d $key $value
- }
- }
- variable menulevels 0
- # some variables for typing-in-menus support
- variable input_buffer ""
- variable input_last_time [openmsx_info realtime]
- variable input_timeout 1; #sec
- proc push_menu_info {} {
- variable menulevels
- incr menulevels 1
- set levelname "menuinfo_$menulevels"
- variable $levelname
- set $levelname [uplevel {dict create \
- name $name lst $lst menu_len $menu_len presentation $presentation \
- menutextexpressions $menutextexpressions menutexts $menutexts \
- selectinfo $selectinfo selectidx $selectidx scrollidx \
- $scrollidx on_close $on_close}]
- }
- proc peek_menu_info {} {
- variable menulevels
- uplevel upvar #0 osd_menu::menuinfo_$menulevels menuinfo
- }
- proc set_selectidx {value} {
- peek_menu_info
- dict set menuinfo selectidx $value
- }
- proc set_scrollidx {value} {
- peek_menu_info
- dict set menuinfo scrollidx $value
- }
- proc menu_create {menudef} {
- variable menulevels
- variable default_bg_color
- variable default_text_color
- variable default_select_color
- variable default_header_text_color
- set name "menu[expr {$menulevels + 1}]"
- set defactions [get_optional menudef "actions" ""]
- set bgcolor [get_optional menudef "bg-color" $default_bg_color]
- set deftextcolor [get_optional menudef "text-color" $default_text_color]
- set selectcolor [get_optional menudef "select-color" $default_select_color]
- set deffontsize [get_optional menudef "font-size" 12]
- set deffont [get_optional menudef "font" "skins/Vera.ttf.gz"]
- set bordersize [get_optional menudef "border-size" 0]
- set on_open [get_optional menudef "on-open" ""]
- set on_close [get_optional menudef "on-close" ""]
- osd create rectangle $name -scaled true -rgba $bgcolor -clip true \
- -borderrgba 0x000000ff -bordersize 0.5
- set y $bordersize
- set selectinfo [list]
- set menutextexpressions [list]
- set menutexts [list]
- foreach itemdef [dict get $menudef items] {
- set selectable [get_optional itemdef "selectable" true]
- incr y [get_optional itemdef "pre-spacing" 0]
- set fontsize [get_optional itemdef "font-size" $deffontsize]
- set font [get_optional itemdef "font" $deffont]
- set textcolor [expr {$selectable
- ? [get_optional itemdef "text-color" $deftextcolor]
- : [get_optional itemdef "text-color" $default_header_text_color]}]
- set actions [get_optional itemdef "actions" ""]
- set on_select [get_optional itemdef "on-select" ""]
- set on_deselect [get_optional itemdef "on-deselect" ""]
- set textid "${name}.item${y}"
- if {[dict exists $itemdef textexpr]} {
- set textexpr [dict get $itemdef textexpr]
- lappend menutextexpressions $textid $textexpr
- } else {
- set text [dict get $itemdef text]
- lappend menutexts $textid $text
- }
- osd create text $textid -font $font -size $fontsize \
- -rgba $textcolor -x $bordersize -y $y
- if {$selectable} {
- set allactions [concat $defactions $actions]
- lappend selectinfo [list $y $fontsize $allactions $on_select $on_deselect]
- }
- incr y $fontsize
- incr y [get_optional itemdef "post-spacing" 0]
- }
- set width [dict get $menudef width]
- set height [expr {$y + $bordersize}]
- set xpos [get_optional menudef "xpos" [expr {(320 - $width) / 2}]]
- set ypos [get_optional menudef "ypos" [expr {(240 - $height) / 2}]]
- osd configure $name -x $xpos -y $ypos -w $width -h $height
- set selw [expr {$width - 2 * $bordersize}]
- osd create rectangle "${name}.selection" -z -1 -rgba $selectcolor \
- -x $bordersize -w $selw
- set lst [get_optional menudef "lst" ""]
- set menu_len [get_optional menudef "menu_len" 0]
- if {[llength $lst] > $menu_len} {
- set startheight 0
- if {[llength $selectinfo] > 0} {
- # there are selectable items. Start the scrollbar
- # at the top of the first selectable item
- # (skipping headers and stuff)
- set startheight [lindex $selectinfo 0 0]
- }
- osd create rectangle "${name}.scrollbar" -z -1 -rgba 0x00000010 \
- -relx 1.0 -x -6 -w 6 -relh 1.0 -h -$startheight -y $startheight -borderrgba 0x00000070 -bordersize 0.5
- osd create rectangle "${name}.scrollbar.thumb" -z -1 -rgba $default_select_color \
- -relw 1.0 -w -2 -x 1
- }
- set presentation [get_optional menudef "presentation" ""]
- set selectidx 0
- set scrollidx 0
- push_menu_info
- uplevel #0 $on_open
- menu_on_select $selectinfo $selectidx
- menu_refresh_top
- menu_update_scrollbar
- }
- proc menu_update_scrollbar {} {
- peek_menu_info
- set name [dict get $menuinfo name]
- if {[osd exists ${name}.scrollbar]} {
- set menu_len [dict get $menuinfo menu_len]
- set scrollidx [dict get $menuinfo scrollidx]
- set selectidx [dict get $menuinfo selectidx]
- set totalitems [llength [dict get $menuinfo lst]]
- set height [expr {1.0*$menu_len/$totalitems}]
- set minheight 0.05 ;# TODO: derive from width of bar
- set height [expr {$height > $minheight ? $height : $minheight}]
- set pos [expr {1.0*($scrollidx+$selectidx)/($totalitems-1)}]
- # scale the pos to the usable range
- set pos [expr {$pos*(1.0-$height)}]
- osd configure "${name}.scrollbar.thumb" -relh $height -rely $pos
- }
- }
- proc menu_refresh_top {} {
- peek_menu_info
- foreach {osdid textexpr} [dict get $menuinfo menutextexpressions] {
- set cmd [list subst $textexpr]
- osd configure $osdid -text [uplevel #0 $cmd]
- }
- foreach {osdid text} [dict get $menuinfo menutexts] {
- osd configure $osdid -text $text
- }
- set selectinfo [dict get $menuinfo selectinfo]
- if {[llength $selectinfo] == 0} return
- set selectidx [dict get $menuinfo selectidx ]
- lassign [lindex $selectinfo $selectidx] sely selh
- osd configure "[dict get $menuinfo name].selection" -y $sely -h $selh
- }
- proc menu_close_top {} {
- variable menulevels
- peek_menu_info
- menu_on_deselect [dict get $menuinfo selectinfo] [dict get $menuinfo selectidx]
- uplevel #0 [dict get $menuinfo on_close]
- osd destroy [dict get $menuinfo name]
- unset menuinfo
- incr menulevels -1
- if {$menulevels == 0} {
- menu_last_closed
- }
- }
- proc menu_close_all {} {
- variable menulevels
- while {$menulevels} {
- menu_close_top
- }
- }
- proc menu_setting {cmd_result} {
- menu_refresh_top
- }
- proc menu_updown {delta} {
- peek_menu_info
- set num [llength [dict get $menuinfo selectinfo]]
- if {$num == 0} return
- set old_idx [dict get $menuinfo selectidx]
- menu_reselect [expr {($old_idx + $delta) % $num}]
- }
- proc menu_reselect {new_idx} {
- peek_menu_info
- set selectinfo [dict get $menuinfo selectinfo]
- set old_idx [dict get $menuinfo selectidx]
- menu_on_deselect $selectinfo $old_idx
- set_selectidx $new_idx
- menu_on_select $selectinfo $new_idx
- menu_refresh_top
- }
- proc menu_on_select {selectinfo selectidx} {
- set on_select [lindex $selectinfo $selectidx 3]
- uplevel #0 $on_select
- }
- proc menu_on_deselect {selectinfo selectidx} {
- set on_deselect [lindex $selectinfo $selectidx 4]
- uplevel #0 $on_deselect
- }
- proc menu_action {button} {
- # for any menu action, clear the input buffer
- variable input_buffer
- set input_buffer ""
- peek_menu_info
- set selectidx [dict get $menuinfo selectidx ]
- menu_action_idx $selectidx $button
- }
- proc menu_action_idx {idx button} {
- peek_menu_info
- set selectinfo [dict get $menuinfo selectinfo]
- set actions [lindex $selectinfo $idx 2]
- set_optional actions UP {osd_menu::menu_updown -1}
- set_optional actions DOWN {osd_menu::menu_updown 1}
- set_optional actions B {osd_menu::menu_close_top}
- set cmd [get_optional actions $button ""]
- uplevel #0 $cmd
- }
- proc get_mouse_coords {} {
- peek_menu_info
- set x 2; set y 2
- catch {
- set name [dict get $menuinfo name]
- lassign [osd info $name -mousecoord] x y
- }
- list $x $y
- }
- proc menu_get_mouse_idx {xy} {
- lassign $xy x y
- if {$x < 0 || 1 < $x || $y < 0 || 1 < $y} {return -1}
- peek_menu_info
- set name [dict get $menuinfo name]
- set yy [expr {$y * [osd info $name -h]}]
- set sel 0
- foreach i [dict get $menuinfo selectinfo] {
- lassign $i y h actions
- if {($y <= $yy) && ($yy < ($y + $h))} {
- return $sel
- }
- incr sel
- }
- return -1
- }
- proc menu_mouse_down {} {
- variable mouse_coord
- variable mouse_idx
- set mouse_coord [get_mouse_coords]
- set mouse_idx [menu_get_mouse_idx $mouse_coord]
- if {$mouse_idx != -1} {
- menu_reselect $mouse_idx
- }
- }
- proc menu_mouse_up {} {
- variable mouse_coord
- variable mouse_idx
- set mouse_coord [get_mouse_coords]
- set mouse_idx [menu_get_mouse_idx $mouse_coord]
- if {$mouse_idx != -1} {
- menu_action_idx $mouse_idx A
- }
- unset mouse_coord
- unset mouse_idx
- }
- proc menu_mouse_wheel {event} {
- lassign $event type1 type2 x y
- if {$y > 0} {
- osd_menu::menu_action UP
- } elseif {$y < 0} {
- osd_menu::menu_action DOWN
- }
- }
- proc menu_mouse_motion {} {
- variable mouse_coord
- variable mouse_idx
- if {![info exists mouse_coord]} return
- set new_mouse_coord [get_mouse_coords]
- set new_idx [menu_get_mouse_idx $new_mouse_coord]
- if {$new_idx != -1 && $new_idx != $mouse_idx} {
- menu_reselect $new_idx
- set mouse_coord $new_mouse_coord
- set mouse_idx $new_idx
- return
- }
- if {$mouse_idx != -1} {
- lassign $mouse_coord old_x old_y
- lassign $new_mouse_coord new_x new_y
- set delta_x [expr {$new_x - $old_x}]
- set delta_y [expr {$new_y - $old_y}]
- if {$delta_y > 0.1} {
- menu_action_idx $mouse_idx DOWN
- set mouse_coord $new_mouse_coord
- return
- } elseif {$delta_y < -0.1} {
- menu_action_idx $mouse_idx UP
- set mouse_coord $new_mouse_coord
- return
- } elseif {$delta_x > 0.1} {
- menu_action_idx $mouse_idx RIGHT
- set mouse_coord $new_mouse_coord
- return
- } elseif {$delta_x < -0.1} {
- menu_action_idx $mouse_idx LEFT
- set mouse_coord $new_mouse_coord
- return
- }
- }
- }
- user_setting create string osd_rom_path "OSD Rom Load Menu Last Known Path" $env(HOME)
- user_setting create string osd_disk_path "OSD Disk Load Menu Last Known Path" $env(HOME)
- user_setting create string osd_tape_path "OSD Tape Load Menu Last Known Path" $env(HOME)
- user_setting create string osd_hdd_path "OSD HDD Load Menu Last Known Path" $env(HOME)
- user_setting create string osd_ld_path "OSD LD Load Menu Last Known Path" $env(HOME)
- if {![file exists $::osd_rom_path] || ![file readable $::osd_rom_path]} {
- # revert to default (should always exist)
- unset ::osd_rom_path
- }
- if {![file exists $::osd_disk_path] || ![file readable $::osd_disk_path]} {
- # revert to default (should always exist)
- unset ::osd_disk_path
- }
- if {![file exists $::osd_tape_path] || ![file readable $::osd_tape_path]} {
- # revert to default (should always exist)
- unset ::osd_tape_path
- }
- if {![file exists $::osd_hdd_path] || ![file readable $::osd_hdd_path]} {
- # revert to default (should always exist)
- unset ::osd_hdd_path
- }
- if {![file exists $::osd_ld_path] || ![file readable $::osd_ld_path]} {
- # revert to default (should always exist)
- unset ::osd_ld_path
- }
- variable taperecordings_directory [file normalize $::env(OPENMSX_USER_DATA)/../taperecordings]
- proc main_menu_open {} {
- do_menu_open [create_main_menu]
- }
- proc do_menu_open {top_menu} {
- variable is_dingux
- # close console, because the menu interferes with it
- set ::console off
- # also remove other OSD controlled widgets (like the osd keyboard)
- if {[info exists ::osd_control::close]} {
- eval $::osd_control::close
- }
- # end tell how to close this widget
- namespace eval ::osd_control {set close ::osd_menu::main_menu_close}
- menu_create $top_menu
- set ::pause true
- # TODO make these bindings easier to customize
- bind -layer osd_menu "OSDcontrol UP PRESS" -repeat {osd_menu::menu_action UP }
- bind -layer osd_menu "OSDcontrol DOWN PRESS" -repeat {osd_menu::menu_action DOWN }
- bind -layer osd_menu "OSDcontrol LEFT PRESS" -repeat {osd_menu::menu_action LEFT }
- bind -layer osd_menu "OSDcontrol RIGHT PRESS" -repeat {osd_menu::menu_action RIGHT}
- bind -layer osd_menu "mouse button1 down" {osd_menu::menu_mouse_down}
- bind -layer osd_menu "mouse button1 up" {osd_menu::menu_mouse_up}
- bind -layer osd_menu "mouse button3 up" {osd_menu::menu_close_top}
- bind -layer osd_menu "mouse motion" {osd_menu::menu_mouse_motion}
- bind -layer osd_menu "mouse wheel" -event {osd_menu::menu_mouse_wheel}
- if {$is_dingux} {
- bind -layer osd_menu "keyb LCTRL" {osd_menu::menu_action A }
- bind -layer osd_menu "keyb LALT" {osd_menu::menu_action B }
- } else {
- bind -layer osd_menu "OSDcontrol A PRESS" {osd_menu::menu_action A }
- bind -layer osd_menu "OSDcontrol B PRESS" {osd_menu::menu_action B }
- # on Android, use BACK button to go back in menus
- bind -layer osd_menu "keyb BACK" {osd_menu::menu_action B }
- }
- bind -layer osd_menu "CTRL+UP" {osd_menu::select_menu_idx 0}
- bind -layer osd_menu "CTRL+LEFT" {osd_menu::select_menu_idx 0}
- bind -layer osd_menu "keyb HOME" {osd_menu::select_menu_idx 0}
- set alphanum {a b c d e f g h i j k l m n o p q r s t u v w x y z 0 1 2 3 4 5 6 7 8 9}
- foreach char $alphanum {
- bind -layer osd_menu "keyb $char" "osd_menu::handle_keyboard_input $char"
- }
- activate_input_layer osd_menu -blocking
- }
- proc main_menu_close {} {
- menu_close_all
- }
- proc main_menu_toggle {} {
- variable menulevels
- if {$menulevels} {
- # there is at least one menu open, close it
- menu_close_all
- } else {
- # none open yet, open main menu
- main_menu_open
- }
- }
- proc menu_last_closed {} {
- set ::pause false
- after realtime 0 {deactivate_input_layer osd_menu}
- namespace eval ::osd_control {unset close}
- }
- proc prepare_menu_list {lst num menudef} {
- set execute [dict get $menudef execute]
- set header [dict get $menudef header]
- set item_extra [get_optional menudef item ""]
- set on_select [get_optional menudef on-select ""]
- set on_deselect [get_optional menudef on-deselect ""]
- set presentation [get_optional menudef presentation $lst]
- # 'assert': presentation should have same length as item list!
- if {[llength $presentation] != [llength $lst]} {
- error "Presentation should be of same length as item list!"
- }
- dict set menudef presentation $presentation
- lappend header "selectable" "false"
- set items [list $header]
- set lst_len [llength $lst]
- set menu_len [expr {$lst_len < $num ? $lst_len : $num}]
- for {set i 0} {$i < $menu_len} {incr i} {
- set actions [list "A" "osd_menu::list_menu_item_exec {$execute} $i"]
- if {$i == 0} {
- lappend actions "UP" "osd_menu::move_selection -1"
- }
- if {$i == ($menu_len - 1)} {
- lappend actions "DOWN" "osd_menu::move_selection 1"
- }
- lappend actions "LEFT" "osd_menu::move_selection -$menu_len"
- lappend actions "RIGHT" "osd_menu::move_selection $menu_len"
- set item [list "textexpr" "\[osd_menu::list_menu_item_show $i\]" \
- "actions" $actions]
- if {$on_select ne ""} {
- lappend item "on-select" "osd_menu::list_menu_item_select $i $on_select"
- }
- if {$on_deselect ne ""} {
- lappend item "on-deselect" "osd_menu::list_menu_item_select $i $on_deselect"
- }
- lappend items [concat $item $item_extra]
- }
- dict set menudef items $items
- dict set menudef lst $lst
- dict set menudef menu_len $menu_len
- return $menudef
- }
- proc list_menu_item_exec {execute pos} {
- peek_menu_info
- {*}$execute [lindex [dict get $menuinfo lst] [expr {$pos + [dict get $menuinfo scrollidx]}]]
- }
- proc list_menu_item_show {pos} {
- peek_menu_info
- return [lindex [dict get $menuinfo presentation] [expr {$pos + [dict get $menuinfo scrollidx]}]]
- }
- proc list_menu_item_select {pos select_proc} {
- peek_menu_info
- $select_proc [lindex [dict get $menuinfo lst] [expr {$pos + [dict get $menuinfo scrollidx]}]]
- }
- proc move_selection {delta} {
- peek_menu_info
- set lst_last [expr {[llength [dict get $menuinfo lst]] - 1}]
- set scrollidx [dict get $menuinfo scrollidx]
- set selectidx [dict get $menuinfo selectidx]
- set old_itemidx [expr {$scrollidx + $selectidx}]
- set new_itemidx [expr {$old_itemidx + $delta}]
- if {$new_itemidx < 0} {
- # Before first element
- if {$old_itemidx == 0} {
- # if first element was already selected, wrap to last
- set new_itemidx $lst_last
- } else {
- # otherwise, clamp to first element
- set new_itemidx 0
- }
- } elseif {$new_itemidx > $lst_last} {
- # After last element
- if {$old_itemidx == $lst_last} {
- # if last element was already selected, wrap to first
- set new_itemidx 0
- } else {
- # otherwise clam to last element
- set new_itemidx $lst_last
- }
- }
- select_menu_idx $new_itemidx
- }
- proc select_menu_idx {itemidx} {
- peek_menu_info
- set menu_len [dict get $menuinfo menu_len]
- set scrollidx [dict get $menuinfo scrollidx]
- set selectidx [dict get $menuinfo selectidx]
- set selectinfo [dict get $menuinfo selectinfo]
- menu_on_deselect $selectinfo $selectidx
- set selectidx [expr {$itemidx - $scrollidx}]
- if {$selectidx < 0} {
- incr scrollidx $selectidx
- set selectidx 0
- } elseif {($menu_len > 0) && ($selectidx >= $menu_len)} {
- set selectidx [expr {$menu_len - 1}]
- set scrollidx [expr {$itemidx - $selectidx}]
- }
- set_selectidx $selectidx
- set_scrollidx $scrollidx
- menu_on_select $selectinfo $selectidx
- menu_refresh_top
- menu_update_scrollbar
- }
- proc select_menu_item {item} {
- peek_menu_info
- set index [lsearch -exact [dict get $menuinfo lst] $item]
- if {$index == -1} return
- select_menu_idx $index
- }
- proc handle_keyboard_input {char} {
- variable input_buffer
- variable input_last_time
- variable input_timeout
- set current_time [openmsx_info realtime]
- if {[expr {$current_time - $input_last_time}] < $input_timeout} {
- set input_buffer "$input_buffer$char"
- } else {
- set input_buffer $char
- }
- set input_last_time $current_time
- osd_menu::select_next_menu_item_starting_with $input_buffer
- }
- proc select_next_menu_item_starting_with {text} {
- peek_menu_info
- set items [dict get $menuinfo presentation]
- if {[llength $items] == 0} return
- set selectidx [dict get $menuinfo selectidx]
- set scrollidx [dict get $menuinfo scrollidx]
- set itemidx [expr {$scrollidx + $selectidx}]
- # start after the current item if this is a new search
- if {[string length $text] == 1} {
- incr itemidx
- }
- # use the list twice to wrap
- set index [lsearch -glob -nocase -start $itemidx [concat $items $items] "$text*"]
- if {$index == -1} return
- set index [expr {$index % [llength $items]}]
- select_menu_idx $index
- }
- variable mediaslot_info [dict create \
- "rom" [dict create mediabasecommand "cart" mediapath "::osd_rom_path" listtype "rom" itemtext "Load ROM" shortmediaslotname "slot" longmediaslotname "cartridge slot"]\
- "disk" [dict create mediabasecommand "disk" mediapath "::osd_disk_path" listtype "disk" itemtext "Insert disk" shortmediaslotname "drive" longmediaslotname "disk drive"]\
- #"cassette" [dict create mediabasecommand "cassetteplayer" mediapath "::osd_tape_path" listtype "tape" itemtext "Set tape" shortmediaslotname "xxx" longmediaslotname "cassette player"]\
- ]
- proc create_slot_actions_to_select_file {slot path listtype} {
- # the action A is checking what is currently in that media slot (e.g. diska) and if it's not empty, it puts that path as last known path. Then it creates the menu and afterwards selects the current item.
- return [list actions [list A "set curSel \[lindex \[$slot\] 1\]; set $path \[expr {\$curSel ne {} ? \[file dirname \$curSel\] : \$$path}\]; osd_menu::menu_create \[osd_menu::menu_create_${listtype}_list \$$path $slot\]; catch { osd_menu::select_menu_item \[file tail \$curSel\]}"]]
- }
- proc get_slot_str {slot} {
- return [string toupper [string index $slot end]]
- }
- proc create_slot_menu_def {slots path listtype menutitle create_action_proc} {
- set menudef {
- font-size 8
- border-size 2
- width 150
- xpos 100
- ypos 80
- }
- lappend items [list text $menutitle\
- font-size 12\
- post-spacing 6\
- selectable false\
- ]
- foreach slot $slots {
- set slotcontent "(empty)"
- if {![string match "empty*" [lindex [$slot] 2]]} {
- set slotcontent [file tail [lindex [$slot] 1]]
- }
- lappend items [list text "[get_slot_str $slot]: $slotcontent" {*}[$create_action_proc $slot $path $listtype]]
- }
- dict set menudef items $items
- return $menudef
- }
- proc create_media_menu_items {file_type_category} {
- variable mediaslot_info
- set longmediaslotname [dict get $mediaslot_info $file_type_category longmediaslotname]
- set shortmediaslotname [dict get $mediaslot_info $file_type_category shortmediaslotname]
- set mediapath [dict get $mediaslot_info $file_type_category mediapath]
- set listtype [dict get $mediaslot_info $file_type_category listtype]
- set itemtext [dict get $mediaslot_info $file_type_category itemtext]
- set mediabasecommand [dict get $mediaslot_info $file_type_category mediabasecommand]
- if {[catch ${mediabasecommand}a]} {; # example: Philips NMS 801 without cart slot, or no diskdrives. TODO: make more generic
- lappend menuitems [list text "(No $longmediaslotname present...)"\
- selectable false\
- text-color 0x808080ff\
- ]
- } else {
- set slots [lsort [info command ${mediabasecommand}?]]; # TODO: make more generic
- if {[llength $slots] <= 2} {
- foreach slot $slots {
- lappend menuitems [list text "${itemtext}... (${shortmediaslotname} [get_slot_str $slot])" {*}[create_slot_actions_to_select_file $slot $mediapath $listtype]]
- }
- } else {
- set slot [lindex $slots 0]
- lappend menuitems [list text "${itemtext}... (${shortmediaslotname} [get_slot_str $slot])" {*}[create_slot_actions_to_select_file $slot $mediapath $listtype]]
- lappend menuitems [list text "${itemtext}... (any ${shortmediaslotname})" actions [list A "osd_menu::menu_create \[osd_menu::create_slot_menu_def \[list $slots\] \{$mediapath\} \{$listtype\} \{Select $longmediaslotname...\} \{create_slot_actions_to_select_file\}\]"]]
- }
- }
- return $menuitems
- }
- #
- # definitions of menus
- #
- proc create_main_menu {} {
- set menu_def {
- font-size 10
- border-size 2
- width 160
- }
- lappend items { textexpr "[openmsx_info version]"
- font-size 12
- post-spacing 6
- selectable false }
- lappend items {*}[create_media_menu_items "rom"]
- lappend items {*}[create_media_menu_items "disk"]
- if {[info command hda] ne ""} {; # only exists when hard disk extension available
- foreach drive [lrange [lsort [info command hd?]] 0 1] {
- set drive_str [string toupper [string index $drive end]]
- lappend items [list text "Change HD/SD image... (drive $drive_str)" \
- actions [list A "set curSel \[lindex \[$drive\] 1\]; set ::osd_hdd_path \[expr {\$curSel ne {} ? \[file dirname \$curSel\] : \$::osd_hdd_path}\]; osd_menu::menu_create \[osd_menu::menu_create_hdd_list \$::osd_hdd_path $drive\]; catch { osd_menu::select_menu_item \[file tail \$curSel\]}"]]
- }
- }
- if {[info command laserdiscplayer] ne ""} {; # only exists on some Pioneers
- lappend items { text "Load LaserDisc..."
- actions { A { set curSel [lindex [laserdiscplayer] 1]; set ::osd_ld_path [expr {$curSel ne {} ? [file dirname $curSel] : $::osd_ld_path}]; osd_menu::menu_create [osd_menu::menu_create_ld_list $::osd_ld_path]; catch { osd_menu::select_menu_item [file tail $curSel]}} }
- }
- }
- if {[catch "machine_info connector cassetteport"]} {; # example: turboR
- lappend items { text "(No cassette port present...)"
- selectable false
- text-color 0x808080ff
- post-spacing 3
- }
- } else {
- lappend items { text "Set Tape..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_tape_list $::osd_tape_path]; catch { osd_menu::select_menu_item [file tail [lindex [cassetteplayer] 1]]}} }
- post-spacing 3 }
- }
- lappend items { text "Save State..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_save_state] }}}
- lappend items { text "Load State..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_load_state] }}
- post-spacing 3 }
- lappend items { text "Hardware..."
- actions { A { osd_menu::menu_create [osd_menu::create_hardware_menu] }}
- post-spacing 3 }
- lappend items { text "Misc Settings..."
- actions { A { osd_menu::menu_create $osd_menu::misc_setting_menu }}}
- lappend items { text "Sound Settings..."
- actions { A { osd_menu::menu_create $osd_menu::sound_setting_menu }}}
- lappend items { text "Video Settings..."
- actions { A { osd_menu::menu_create [osd_menu::create_video_setting_menu] }}
- post-spacing 3 }
- lappend items { text "Advanced..."
- actions { A { osd_menu::menu_create $osd_menu::advanced_menu }}
- post-spacing 10 }
- lappend items { text "Reset MSX"
- actions { A { reset; osd_menu::menu_close_all }}}
- lappend items { text "Exit openMSX"
- actions { A quitmenu::quit_menu }}
- dict set menu_def items $items
- return $menu_def
- }
- set misc_setting_menu {
- font-size 8
- border-size 2
- width 160
- xpos 100
- ypos 120
- items {{ text "Misc Settings"
- font-size 10
- post-spacing 6
- selectable false }
- { textexpr "Speed: ${speed}%"
- actions { LEFT { osd_menu::menu_setting [incr speed -1] }
- RIGHT { osd_menu::menu_setting [incr speed 1] }}}
- { textexpr "Fastforward speed: ${fastforwardspeed}%"
- actions { LEFT { osd_menu::menu_setting [incr fastforwardspeed -1] }
- RIGHT { osd_menu::menu_setting [incr fastforwardspeed 1] }}}
- { textexpr "Full speed when loading: [osd_menu::boolean_to_text $fullspeedwhenloading]"
- actions { LEFT { osd_menu::menu_setting [cycle_back fullspeedwhenloading] }
- RIGHT { osd_menu::menu_setting [cycle fullspeedwhenloading] }}}
- { textexpr "Keyboard mapping mode: $kbd_mapping_mode"
- actions { LEFT { osd_menu::menu_setting [cycle_back kbd_mapping_mode] }
- RIGHT { osd_menu::menu_setting [cycle kbd_mapping_mode] }}}
- }}
- set resampler_desc [dict create fast "fast (but low quality)" blip "blip (good speed/quality)" hq "hq (best but uses more CPU)"]
- set sound_setting_menu {
- font-size 8
- border-size 2
- width 180
- xpos 100
- ypos 120
- items {{ text "Sound Settings"
- font-size 10
- post-spacing 6
- selectable false }
- { textexpr "Volume: $master_volume"
- actions { LEFT { osd_menu::menu_setting [incr master_volume -5] }
- RIGHT { osd_menu::menu_setting [incr master_volume 5] }}}
- { textexpr "Mute: [osd_menu::boolean_to_text $mute]"
- actions { LEFT { osd_menu::menu_setting [cycle_back mute] }
- RIGHT { osd_menu::menu_setting [cycle mute] }}}
- { text "Individual Sound Device Settings..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_sound_device_list]}}}
- { textexpr "Resampler: [osd_menu::get_resampler_presentation $resampler]"
- actions { LEFT { osd_menu::menu_setting [cycle_back resampler] }
- RIGHT { osd_menu::menu_setting [cycle resampler] }}}}}
- set horizontal_stretch_desc [dict create 320.0 "none (large borders)" 288.0 "a bit more than all border pixels" 284.0 "all border pixels" 280.0 "a bit less than all border pixels" 272.0 "realistic" 256.0 "no borders at all"]
- proc menu_create_sound_device_list {} {
- set menu_def {
- execute menu_sound_device_select_exec
- font-size 8
- border-size 2
- width 200
- xpos 110
- ypos 130
- header { text "Select Sound Chip"
- font-size 10
- post-spacing 6 }}
- set items [machine_info sounddevice]
- return [prepare_menu_list $items 5 $menu_def]
- }
- proc menu_sound_device_select_exec {item} {
- menu_create [create_sound_device_settings_menu $item]
- select_menu_item $item
- }
- proc create_sound_device_settings_menu {device} {
- set ypos 140
- set menu_def [list \
- font-size 8 \
- border-size 2 \
- width 210 \
- xpos 120 \
- ypos $ypos]
- lappend items [list text "$device Settings" \
- font-size 10 \
- post-spacing 6 \
- selectable false]
- # volume and balance
- foreach aspect [list volume balance] {
- set var_name ::${device}_${aspect}
- set item [list]
- lappend item "textexpr"
- set first [string range $aspect 0 0]
- set rest [string range $aspect 1 end]
- set first [string toupper $first]
- set capped_aspect "${first}${rest}"
- lappend item "$capped_aspect: \[[list set $var_name]]"
- lappend item "actions"
- set actions [list]
- lappend actions "LEFT"
- lappend actions "osd_menu::menu_setting \[[list incr $var_name -5]]"
- lappend actions "RIGHT"
- lappend actions "osd_menu::menu_setting \[[list incr $var_name 5]]"
- lappend item $actions
- lappend items $item
- }
- # channel mute
- set channel_count [soundchip_utils::get_num_channels $device]
- for {set channel 1} {$channel <= $channel_count} {incr channel} {
- set chmute_var_name ${device}_ch${channel}_mute
- set item [list]
- lappend item "textexpr"
- set pretext ""
- if {$channel_count > 1} {
- set pretext "Channel $channel "
- }
- lappend item "${pretext}Mute: \[osd_menu::boolean_to_text \[[list set $chmute_var_name]]]"
- lappend item "actions"
- set actions [list]
- lappend actions "LEFT"
- lappend actions "osd_menu::menu_setting \[[list cycle_back $chmute_var_name]]"
- lappend actions "RIGHT"
- lappend actions "osd_menu::menu_setting \[[list cycle $chmute_var_name]]"
- lappend item $actions
- lappend items $item
- }
- # adjust menu position for longer lists
- # TODO: make this less magic
- if {$channel_count > 8} {;# more won't fit
- dict set menu_def ypos [expr {$ypos - round(($channel_count - 8) * ($ypos - 10)/16)}]
- }
- dict set menu_def items $items
- return $menu_def
- }
- proc boolean_to_text {boolean} {
- if {[lsearch -nocase [list "true" "on" "yes"] $boolean] != -1} {
- return "on"
- } elseif {[lsearch -nocase [list "false" "off" "no"] $boolean] != -1} {
- return "off"
- } else {
- puts stderr "Couldn't make head or tails of: XXX${boolean}XXX"
- return $boolean
- }
- }
- proc create_video_setting_menu {} {
- variable scaling_available
- set menu_def {
- font-size 8
- border-size 2
- width 210
- xpos 100
- ypos 110
- }
- lappend items { text "Video Settings"
- font-size 10
- post-spacing 6
- selectable false }
- if {[expr {[lindex [lindex [openmsx_info setting videosource] 2] 1] > 1}]} {
- lappend items { textexpr "Video source: $videosource"
- actions { LEFT { osd_menu::menu_setting [cycle_back videosource] }
- RIGHT { osd_menu::menu_setting [cycle videosource] }}
- post-spacing 3}
- }
- if {$scaling_available} {
- lappend items { textexpr "Scaler: $scale_algorithm"
- actions { LEFT { osd_menu::menu_setting [cycle_back scale_algorithm] }
- RIGHT { osd_menu::menu_setting [cycle scale_algorithm] }}}
- # only add scale factor setting if it can actually be changed
- set scale_minmax [lindex [openmsx_info setting scale_factor] 2]
- if {[expr {[lindex $scale_minmax 0] != [lindex $scale_minmax 1]}]} {
- lappend items { textexpr "Scale Factor: ${scale_factor}x"
- actions { LEFT { osd_menu::menu_setting [incr scale_factor -1] }
- RIGHT { osd_menu::menu_setting [incr scale_factor 1] }}}
- }
- }
- lappend items { textexpr "Horizontal Stretch: [osd_menu::get_horizontal_stretch_presentation $horizontal_stretch]"
- actions { A { osd_menu::menu_create [osd_menu::menu_create_stretch_list]; osd_menu::select_menu_item $horizontal_stretch }}
- post-spacing 3 }
- if {$::renderer eq "SDLGL-PP"} {
- lappend items { textexpr "VSync: [osd_menu::boolean_to_text $vsync]"
- actions { LEFT { osd_menu::menu_setting [cycle_back vsync] }
- RIGHT { osd_menu::menu_setting [cycle vsync] }}
- post-spacing 3 }
- }
- if {$scaling_available} {
- lappend items { textexpr "Scanline: $scanline%"
- actions { LEFT { osd_menu::menu_setting [incr scanline -1] }
- RIGHT { osd_menu::menu_setting [incr scanline 1] }}}
- lappend items { textexpr "Blur: $blur%"
- actions { LEFT { osd_menu::menu_setting [incr blur -1] }
- RIGHT { osd_menu::menu_setting [incr blur 1] }}}
- }
- if {$::renderer eq "SDLGL-PP"} {
- lappend items { textexpr "Glow: $glow%"
- actions { LEFT { osd_menu::menu_setting [incr glow -1] }
- RIGHT { osd_menu::menu_setting [incr glow 1] }}}
- lappend items { textexpr "Display Deform: $display_deform"
- actions { LEFT { osd_menu::menu_setting [cycle_back display_deform] }
- RIGHT { osd_menu::menu_setting [cycle display_deform] }}}
- }
- lappend items { textexpr "Noise: $noise%"
- actions { LEFT { osd_menu::menu_setting [set noise [expr $noise - 1]] }
- RIGHT { osd_menu::menu_setting [set noise [expr $noise + 1]] }}
- post-spacing 3}
- lappend items { textexpr "Enforce VDP Sprites-per-line Limit: [osd_menu::boolean_to_text $limitsprites]"
- actions { LEFT { osd_menu::menu_setting [cycle_back limitsprites] }
- RIGHT { osd_menu::menu_setting [cycle limitsprites] }}}
- dict set menu_def items $items
- return $menu_def
- }
- proc create_hardware_menu {} {
- set menu_def {
- font-size 8
- border-size 2
- width 175
- xpos 100
- ypos 120
- }
- lappend items { text "Hardware"
- font-size 10
- post-spacing 6
- selectable false }
- lappend items { text "Change Machine..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_load_machine_list]; catch { osd_menu::select_menu_item [machine_info config_name]} }}}
- lappend items { text "Set Current Machine as Default"
- actions { A { set ::default_machine [machine_info config_name]; osd_menu::menu_close_top }}}
- lappend items { text "Extensions..."
- actions { A { osd_menu::menu_create $osd_menu::extensions_menu }}}
- lappend items { text "Connectors..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_connectors_list] }}}
- if {![catch {openmsx_info setting firmwareswitch}]} {
- lappend items { textexpr "Firmware switch: [osd_menu::boolean_to_text $::firmwareswitch]"
- actions { LEFT { osd_menu::menu_setting [cycle_back firmwareswitch] }
- RIGHT { osd_menu::menu_setting [cycle firmwareswitch] }}}
- }
- dict set menu_def items $items
- return $menu_def
- }
- set extensions_menu {
- font-size 8
- border-size 2
- width 175
- xpos 100
- ypos 120
- items {{ text "Extensions"
- font-size 10
- post-spacing 6
- selectable false }
- { text "Add..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_extensions_list] }}}
- { text "Remove..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_plugged_extensions_list] }}}}}
- set advanced_menu {
- font-size 8
- border-size 2
- width 175
- xpos 100
- ypos 120
- items {{ text "Advanced"
- font-size 10
- post-spacing 6
- selectable false }
- { text "Manage Running Machines..."
- actions { A { osd_menu::menu_create $osd_menu::running_machines_menu }}}
- { text "Toys and Utilities..."
- actions { A { osd_menu::menu_create [osd_menu::menu_create_toys_list] }}}}}
- set running_machines_menu {
- font-size 8
- border-size 2
- width 175
- xpos 100
- ypos 120
- items {{ text "Manage Running Machines"
- font-size 10
- post-spacing 6
- selectable false }
- { textexpr "Select Running Machine Tab: [utils::get_machine_display_name]"
- actions { A { osd_menu::menu_create [osd_menu::menu_create_running_machine_list] }}}
- { text "New Running Machine Tab"
- actions { A { osd_menu::menu_create [osd_menu::menu_create_load_machine_list "add"] }}}
- { text "Close Current Machine Tab"
- actions { A { set old_active_machine [activate_machine]; cycle_machine; delete_machine $old_active_machine }}}}}
- proc menu_create_running_machine_list {} {
- set menu_def {
- execute menu_machine_tab_select_exec
- font-size 8
- border-size 2
- width 200
- xpos 110
- ypos 130
- header { text "Select Running Machine"
- font-size 10
- post-spacing 6 }}
- set items [utils::get_ordered_machine_list]
- set presentation [list]
- foreach i $items {
- if {[activate_machine] eq $i} {
- set postfix_text "current"
- } else {
- set postfix_text [utils::get_machine_time $i]
- }
- lappend presentation [format "%s (%s)" [utils::get_machine_display_name ${i}] $postfix_text]
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 5 $menu_def]
- }
- proc menu_machine_tab_select_exec {item} {
- menu_close_top
- activate_machine $item
- }
- proc get_resampler_presentation { value } {
- if {[dict exists $osd_menu::resampler_desc $value]} {
- return [dict get $osd_menu::resampler_desc $value]
- } else {
- return $value
- }
- }
- proc get_horizontal_stretch_presentation { value } {
- if {[dict exists $osd_menu::horizontal_stretch_desc $value]} {
- return [dict get $osd_menu::horizontal_stretch_desc $value]
- } else {
- return "custom: $::horizontal_stretch"
- }
- }
- proc menu_create_stretch_list {} {
- set menu_def [list \
- execute menu_stretch_exec \
- font-size 8 \
- border-size 2 \
- width 150 \
- xpos 110 \
- ypos 130 \
- header { text "Select Horizontal Stretch:"
- font-size 10
- post-spacing 6 }]
- set items [list]
- set presentation [list]
- set values [dict keys $osd_menu::horizontal_stretch_desc]
- if {$::horizontal_stretch ni $values} {
- lappend values $::horizontal_stretch
- }
- foreach value $values {
- lappend items $value
- lappend presentation [osd_menu::get_horizontal_stretch_presentation $value]
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 6 $menu_def]
- }
- proc menu_stretch_exec {value} {
- set ::horizontal_stretch $value
- menu_close_top
- menu_refresh_top
- }
- # Returns list of machines/extensions, but try to filter out duplicates caused
- # by symlinks (e.g. turbor.xml -> Panasonic_FS-A1GT.xml). What this does not
- # catch is a symlink in the systemdir (so link also pointing to the systemdir)
- # and a similarly named file in the userdir. This situation does occur on my
- # development setup, but it shouldn't happen for regular users.
- proc get_filtered_configs {type} {
- set result [list]
- set configs [list]
- foreach t [openmsx_info $type] {
- # try both <name>.xml and <name>/hardwareconfig.xml
- set conf [data_file $type/$t.xml]
- if {![file exists $conf]} {
- set conf [data_file $type/$t/hardwareconfig.xml]
- }
- # follow symlink (on platforms that support links)
- catch {
- set conf [file join [file dirname $conf] [file readlink $conf]]
- }
- # only add if the (possibly resolved link) hasn't been seen before
- if {$conf ni $configs} {
- lappend configs $conf
- lappend result $t
- }
- }
- return $result
- }
- proc menu_create_load_machine_list {{mode "replace"}} {
- if {$mode eq "replace"} {
- set proc_to_exec osd_menu::menu_load_machine_exec_replace
- } elseif {$mode eq "add"} {
- set proc_to_exec osd_menu::menu_load_machine_exec_add
- } else {
- error "Undefined mode: $mode"
- }
- set menu_def [list \
- execute $proc_to_exec \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 110 \
- ypos 130 \
- header { text "Select Machine to Run"
- font-size 10
- post-spacing 6 }]
- set items [get_filtered_configs machines]
- foreach i $items {
- set extra_info ""
- if {$i eq $::default_machine} {
- set extra_info " (default)"
- }
- lappend presentation "[utils::get_machine_display_name_by_config_name $i]$extra_info"
- }
- set items_sorted [list]
- set presentation_sorted [list]
- foreach i [lsort -dictionary -indices $presentation] {
- lappend presentation_sorted [lindex $presentation $i]
- lappend items_sorted [lindex $items $i]
- }
- lappend menu_def presentation $presentation_sorted
- return [prepare_menu_list $items_sorted 10 $menu_def]
- }
- proc menu_load_machine_exec_replace {item} {
- if {[catch {machine $item} errorText]} {
- osd::display_message $errorText error
- } else {
- menu_close_all
- }
- }
- proc menu_load_machine_exec_add {item} {
- set id [create_machine]
- set err [catch {${id}::load_machine $item} error_result]
- if {$err} {
- delete_machine $id
- osd::display_message "Error starting [utils::get_machine_display_name_by_config_name $item]: $error_result" error
- } else {
- menu_close_top
- activate_machine $id
- }
- }
- proc menu_create_extensions_list {} {
- set menu_def {
- execute menu_add_extension_exec
- font-size 8
- border-size 2
- width 200
- xpos 110
- ypos 130
- header { text "Select Extension to Add"
- font-size 10
- post-spacing 6 }}
- set items [get_filtered_configs extensions]
- set presentation [list]
- foreach i $items {
- lappend presentation [utils::get_extension_display_name_by_config_name $i]
- }
- set items_sorted [list]
- set presentation_sorted [list]
- foreach i [lsort -dictionary -indices $presentation] {
- lappend presentation_sorted [lindex $presentation $i]
- lappend items_sorted [lindex $items $i]
- }
- lappend menu_def presentation $presentation_sorted
- return [prepare_menu_list $items_sorted 10 $menu_def]
- }
- proc menu_add_extension_exec {item} {
- if {[catch {ext $item} errorText]} {
- osd::display_message $errorText error
- } else {
- menu_close_all
- }
- }
- proc menu_create_plugged_extensions_list {} {
- set menu_def {
- execute menu_remove_extension_exec
- font-size 8
- border-size 2
- width 200
- xpos 110
- ypos 130
- header { text "Select Extension to Remove"
- font-size 10
- post-spacing 6 }}
- set items [list_extensions]
- set possible_items [get_filtered_configs extensions]
- set useful_items [list]
- foreach item $items {
- if {$item in $possible_items} {
- lappend useful_items $item
- }
- }
- set presentation [list]
- foreach i $useful_items {
- lappend presentation [utils::get_extension_display_name_by_config_name ${i}]
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $useful_items 10 $menu_def]
- }
- proc menu_remove_extension_exec {item} {
- menu_close_all
- remove_extension $item
- }
- proc menu_create_connectors_list {} {
- set menu_def {
- execute menu_connector_exec
- font-size 8
- border-size 2
- width 200
- xpos 100
- ypos 120
- header { text "Connectors"
- font-size 10
- post-spacing 6 }}
- set items [machine_info connector]
- set presentation [list]
- foreach item $items {
- set plugged [get_pluggable_for_connector $item]
- set plugged_presentation ""
- if {$plugged ne ""} {
- set plugged_presentation " ([machine_info pluggable $plugged])"
- }
- lappend presentation "[machine_info connector $item]: $plugged$plugged_presentation"
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 5 $menu_def]
- }
- proc menu_connector_exec {item} {
- menu_create [create_menu_pluggable_list $item]
- select_menu_item [get_pluggable_for_connector $item]
- }
- proc create_menu_pluggable_list {connector} {
- set menu_def [list \
- execute [list menu_plug_exec $connector] \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 110 \
- ypos 140 \
- header [list text "What to Plug into [machine_info connector $connector]?" \
- font-size 10 \
- post-spacing 6 ]]
- set items [list]
- set class [machine_info connectionclass $connector]
- # find out which pluggables are already plugged
- # (currently a pluggable can be used only once per machine)
- set already_plugged [list]
- foreach other_connector [machine_info connector] {
- set other_plugged [get_pluggable_for_connector $other_connector]
- if {$other_plugged ne "" && $other_connector ne $connector} {
- lappend already_plugged $other_plugged
- }
- }
- # get a list of all pluggables that fit this connector
- # and which are not plugged yet in other connectors
- foreach pluggable [machine_info pluggable] {
- if {$pluggable ni $already_plugged && [machine_info connectionclass $pluggable] eq $class} {
- lappend items $pluggable
- }
- }
- set presentation [list]
- foreach item $items {
- lappend presentation "$item: [machine_info pluggable $item]"
- }
- set plugged [get_pluggable_for_connector $connector]
- if {$plugged ne ""} {
- set items [linsert $items 0 "--unplug--"]
- set presentation [linsert $presentation 0 "Nothing, unplug $plugged ([machine_info pluggable $plugged])"]
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 5 $menu_def]
- }
- proc menu_plug_exec {connector pluggable} {
- set command ""
- if {$pluggable eq "--unplug--"} {
- set command "unplug {$connector}"
- } else {
- set command "plug {$connector} {$pluggable}"
- }
- #note: NO braces around $command
- if {[catch $command errorText]} {
- osd::display_message $errorText error
- } else {
- menu_close_top
- # refresh the connectors menu
- # The list must be recreated, so menu_refresh_top won't work
- menu_close_top
- menu_create [menu_create_connectors_list]
- }
- }
- proc menu_create_toys_list {} {
- set menu_def {
- execute menu_toys_exec
- font-size 8
- border-size 2
- width 200
- xpos 100
- ypos 120
- header { text "Toys and Utilities"
- font-size 10
- post-spacing 6 }}
- set items [list]
- set presentation [list]
- # This also picks up 'lazy' command names
- foreach cmd [openmsx::all_command_names] {
- if {[string match toggle_* $cmd]} {
- lappend items $cmd
- lappend presentation [string map {_ " "} [string range $cmd 7 end]]
- }
- }
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 5 $menu_def]
- }
- proc menu_toys_exec {toy} {
- return [$toy]
- }
- proc ls {directory extensions} {
- set dirs [list]
- set specialdir [list]
- set items [list]
- if {[catch {
- set files [glob -nocomplain -tails -directory $directory -types {f r} *]
- set items [lsearch -regexp -all -inline -nocase $files .*\\.($extensions)]
- set dirs [glob -nocomplain -tails -directory $directory -types {d r x} *]
- set specialdir [glob -nocomplain -tails -directory $directory -types {hidden d} ".openMSX"]
- } errorText]} {
- osd::display_message "Unable to read dir $directory: $errorText" error
- }
- set dirs2 [list]
- foreach dir [concat $dirs $specialdir] {
- lappend dirs2 "$dir/"
- }
- set extra_entries [list]
- set volumes [file volumes]
- if {$directory ni $volumes} {
- # check whether .. is readable (it's not always so on Android)
- if {[file readable [file join $directory ..]]} {
- lappend extra_entries ".."
- }
- } else {
- if {[llength $volumes] > 1} {
- set extra_entries $volumes
- }
- }
- return [concat [lsort $extra_entries] [lsort $dirs2] [lsort $items]]
- }
- proc is_empty_dir {directory extensions} {
- set files [list]
- catch {set files [glob -nocomplain -tails -directory $directory -types {f r} *]}
- set items [lsearch -regexp -all -inline -nocase $files .*\\.($extensions)]
- if {[llength $items] != 0} {return false}
- set dirs [list]
- catch {set dirs [glob -nocomplain -tails -directory $directory -types {d r x} *]}
- if {[llength $dirs] != 0} {return false}
- set specialdir [list]
- catch {set specialdir [glob -nocomplain -tails -directory $directory -types {hidden d} ".openMSX"]}
- if {[llength $specialdir] != 0} {return false}
- return true
- }
- proc menu_create_rom_list {path slot} {
- set menu_def [list execute [list menu_select_rom $slot] \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 100 \
- ypos 120 \
- header { textexpr "ROMs $::osd_rom_path" \
- font-size 10 \
- post-spacing 6 }]
- set extensions "rom|ri|mx1|mx2|zip|gz"
- set items [list]
- set presentation [list]
- if {[lindex [$slot] 2] ne "empty"} {
- lappend items "--eject--"
- lappend presentation "--eject-- [file tail [lindex [$slot] 1]]"
- }
- set i 1
- foreach pool_path [filepool::get_paths_for_type rom] {
- if {$path ne $pool_path && [file exists $pool_path] &&
- ![is_empty_dir $pool_path $extensions]} {
- lappend items $pool_path
- lappend presentation "\[ROM Pool $i\]"
- }
- incr i
- }
- set files [ls $path $extensions]
- set items [concat $items $files]
- set presentation [concat $presentation $files]
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_select_rom {slot item {open_main false}} {
- if {$item eq "--eject--"} {
- menu_close_all
- $slot eject
- reset
- } else {
- set fullname [file join $::osd_rom_path $item]
- if {[file isdirectory $fullname]} {
- menu_close_top
- set ::osd_rom_path [file normalize $fullname]
- menu_create [menu_create_rom_list $::osd_rom_path $slot]
- } else {
- set mappertype ""
- set hash [sha1sum $fullname]
- if {[catch {set mappertype [dict get [openmsx_info software $hash] mapper_type_name]}]} {
- # not in the database, execute after selecting mapper type
- set menu_proc menu_create
- if {$open_main} { set menu_proc do_menu_open }
- ${menu_proc} [menu_create_mappertype_list $slot $fullname]
- } else {
- # in the database, so just execute
- menu_rom_with_mappertype_exec $slot $fullname $mappertype
- }
- }
- }
- }
- proc menu_rom_with_mappertype_exec {slot fullname mappertype} {
- if {[catch {$slot $fullname -romtype $mappertype} errorText]} {
- osd::display_message "Can't insert ROM: $errorText" error
- } else {
- menu_close_all
- set rominfo [getlist_rom_info]
- set message1 "Now running ROM"
- set message2 " \n"
- dict with rominfo {
- if {$slotname ne "" && $slot ne ""} {
- append message1 " in cartridge slot $slotname (slot $slot)"
- }
- append message1 ":\n"
- if {$title ne ""} {
- append message1 "Title:\n"
- append message2 "$title\n"
- } elseif {$filename ne ""} {
- append message1 "File:\n"
- append message2 "$filename\n"
- }
- if {$year ne ""} {
- append message1 "Year:\n"
- append message2 "$year\n"
- }
- if {$company ne ""} {
- append message1 "Company:\n"
- append message2 "$company\n"
- }
- if {$country ne ""} {
- append message1 "Country:\n"
- append message2 "$country\n"
- }
- if {$status ne ""} {
- append message1 "Status:\n"
- append message2 "$status\n"
- }
- if {$remark ne ""} {
- append message1 "Remark:\n"
- append message2 "$remark\n"
- }
- if {$mappertype ne ""} {
- append message1 "Mapper:\n"
- append message2 "$mappertype\n"
- }
- }
- osd::display_message $message1
- set txt_size 6
- set xpos 35
- # TODO: prevent this from being duplicated from osd_widgets::text_box
- if {$::scale_factor == 1} {
- set txt_size 9
- set xpos 53
- }
- # TODO: this code knows the internal name of the widget of osd::display_message proc... it shouldn't need to.
- osd create text osd_display_message.rominfo_text -x $xpos -y 2 -size $txt_size -rgba 0xffffffff -text "$message2"
- reset
- }
- }
- proc menu_create_mappertype_list {slot fullname} {
- set menu_def [list execute [list menu_rom_with_mappertype_exec $slot $fullname] \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 100 \
- ypos 120 \
- header { text "Select mapper type" \
- font-size 10 \
- post-spacing 6 }]
- set items [openmsx_info romtype]
- set presentation [list]
- foreach i $items {
- lappend presentation "[dict get [openmsx_info romtype $i] description]"
- }
- set items_sorted [list "auto"]
- set presentation_sorted [list "Auto-detect (guess)"]
- foreach i [lsort -dictionary -indices $presentation] {
- lappend presentation_sorted [lindex $presentation $i]
- lappend items_sorted [lindex $items $i]
- }
- lappend menu_def presentation $presentation_sorted
- return [prepare_menu_list $items_sorted 10 $menu_def]
- }
- proc menu_create_disk_list {path drive} {
- set menu_def [list execute [list menu_select_disk $drive] \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 100 \
- ypos 120 \
- header { textexpr "Disks $::osd_disk_path" \
- font-size 10 \
- post-spacing 6 }]
- set cur_image [lindex [$drive] 1]
- set extensions "dsk|zip|gz|xsa|dmk|di1|di2|fd?|1|2|3|4|5|6|7|8|9"
- set items [list]
- set presentation [list]
- if {[lindex [$drive] 2] ne "empty readonly"} {
- lappend items "--eject--"
- lappend presentation "--eject-- [file tail $cur_image]"
- }
- set i 1
- foreach pool_path [filepool::get_paths_for_type disk] {
- if {$path ne $pool_path && [file exists $pool_path] &&
- ![is_empty_dir $pool_path $extensions]} {
- lappend items $pool_path
- lappend presentation "\[Disk Pool $i\]"
- }
- incr i
- }
- if {$cur_image ne $path} {
- lappend items "."
- lappend presentation "--insert this dir as disk--"
- }
- set files [ls $path $extensions]
- set items [concat $items $files]
- set presentation [concat $presentation $files]
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_select_disk {drive item {dummy false}} {
- if {$item eq "--eject--"} {
- set cur_image [lindex [$drive] 1]
- menu_close_all
- $drive eject
- osd::display_message "Disk $cur_image ejected from drive [get_slot_str $drive]!"
- } else {
- # if the item is already a directory, it's an absolute path, use that as fullname
- if {[file isdirectory $item] && $item ne "." && $item ne ".." && $item ni [file volumes]} {
- set fullname $item
- set abspath true
- } else {
- set fullname [file normalize [file join $::osd_disk_path $item]]
- set abspath false
- }
- if {[file isdirectory $fullname] && $item ne "." && !$abspath} {
- menu_close_top
- set ::osd_disk_path [file normalize $fullname]
- menu_create [menu_create_disk_list $::osd_disk_path $drive]
- } else {
- if {[catch {$drive $fullname} errorText]} {
- osd::display_message "Can't insert disk: $errorText" error
- } else {
- menu_close_all
- if {$item eq "."} { set item $fullname }
- osd::display_message "Disk $item inserted in drive [get_slot_str $drive]!"
- }
- }
- }
- }
- proc menu_create_tape_list {path} {
- variable taperecordings_directory
- set menu_def { execute menu_select_tape
- font-size 8
- border-size 2
- width 200
- xpos 100
- ypos 120
- header { textexpr "Tapes $::osd_tape_path"
- font-size 10
- post-spacing 6 }}
- set extensions "cas|wav|zip|gz"
- set items [list]
- set presentation [list]
- lappend items "--create--"
- lappend presentation "--create new and insert--"
- set inserted [lindex [cassetteplayer] 1]
- if {$inserted ne ""} {
- lappend items "--eject--"
- lappend presentation "--eject-- [file tail $inserted]"
- lappend items "--rewind--"
- lappend presentation "--rewind-- [file tail $inserted]"
- }
- if {$path ne $taperecordings_directory && [file exists $taperecordings_directory]} {
- lappend items $taperecordings_directory
- lappend presentation "\[My Tape Recordings\]"
- }
- set i 1
- foreach pool_path [filepool::get_paths_for_type tape] {
- if {$path ne $pool_path && [file exists $pool_path] &&
- ![is_empty_dir $pool_path $extensions]} {
- lappend items $pool_path
- lappend presentation "\[Tape Pool $i\]"
- }
- incr i
- }
- set files [ls $path $extensions]
- set items [concat $items $files]
- set presentation [concat $presentation $files]
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_select_tape {item} {
- variable taperecordings_directory
- if {$item eq "--create--"} {
- menu_close_all
- osd::display_message [cassetteplayer new [menu_free_tape_name]]
- } elseif {$item eq "--eject--"} {
- menu_close_all
- osd::display_message [cassetteplayer eject]
- } elseif {$item eq "--rewind--"} {
- menu_close_all
- osd::display_message [cassetteplayer rewind]
- } else {
- set fullname [file join $::osd_tape_path $item]
- if {[file isdirectory $fullname]} {
- menu_close_top
- set ::osd_tape_path [file normalize $fullname]
- menu_create [menu_create_tape_list $::osd_tape_path]
- } else {
- if {[catch {cassetteplayer $fullname} errorText]} {
- osd::display_message "Can't set tape: $errorText" error
- } else {
- osd::display_message "Inserted tape $item!"
- menu_close_all
- }
- }
- }
- }
- proc menu_free_tape_name {} {
- variable taperecordings_directory
- set existing [list]
- foreach f [lsort [glob -tails -directory $taperecordings_directory -type f -nocomplain *.wav]] {
- lappend existing [file rootname $f]
- }
- set i 1
- while 1 {
- set name [format "[guess_title untitled] %04d" $i]
- if {$name ni $existing} {
- return $name
- }
- incr i
- }
- }
- proc menu_create_hdd_list {path drive} {
- return [prepare_menu_list [ls $path "dsk|zip|gz|hdd"] \
- 10 \
- [list execute [list menu_select_hdd $drive]\
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 100 \
- ypos 120 \
- header { textexpr "Hard disk images $::osd_hdd_path"
- font-size 10
- post-spacing 6 }]]
- }
- proc menu_select_hdd {drive item} {
- set fullname [file join $::osd_hdd_path $item]
- if {[file isdirectory $fullname]} {
- menu_close_top
- set ::osd_hdd_path [file normalize $fullname]
- menu_create [menu_create_hdd_list $::osd_hdd_path $drive]
- } else {
- confirm_action "Really power off to change HDD image?" osd_menu::confirm_change_hdd [list $item $drive]
- }
- }
- proc confirm_change_hdd {item result} {
- menu_close_top
- if {$result eq "Yes"} {
- set fullname [file join $::osd_hdd_path [lindex $item 0]]
- if {[catch {set ::power off; [lindex $item 1] $fullname} errorText]} {
- osd::display_message "Can't change hard disk image: $errorText" error
- # TODO: we already powered off even though the file may be invalid... save state first?
- } else {
- osd::display_message "Changed hard disk image to [lindex $item 0]!"
- menu_close_all
- }
- set ::power on
- }
- }
- proc menu_create_ld_list {path} {
- set menu_def [list execute menu_select_ld \
- font-size 8 \
- border-size 2 \
- width 200 \
- xpos 100 \
- ypos 120 \
- header { textexpr "LaserDiscs $::osd_ld_path" \
- font-size 10 \
- post-spacing 6 }]
- set cur_image [lindex [laserdiscplayer] 1]
- set extensions "ogv"
- set items [list]
- set presentation [list]
- if {$cur_image ne ""} {
- lappend items "--eject--"
- lappend presentation "--eject-- [file tail $cur_image]"
- }
- set files [ls $path $extensions]
- set items [concat $items $files]
- set presentation [concat $presentation $files]
- lappend menu_def presentation $presentation
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_select_ld {item} {
- if {$item eq "--eject--"} {
- menu_close_all
- osd::display_message [laserdiscplayer eject]
- set cur_image [lindex [laserdiscplayer] 1]
- osd::display_message "LaserDisc $cur_image ejected!"
- } else {
- set fullname [file join $::osd_ld_path $item]
- if {[file isdirectory $fullname]} {
- menu_close_top
- set ::osd_ld_path [file normalize $fullname]
- menu_create [menu_create_ld_list $::osd_ld_path]
- } else {
- if {[catch {laserdiscplayer insert $fullname} errorText]} {
- osd::display_message "Can't load LaserDisc: $errorText" error
- } else {
- osd::display_message "Loaded LaserDisc $item!"
- menu_close_all
- }
- }
- }
- }
- proc get_savestates_list_presentation_sorted {} {
- set presentation [list]
- foreach i [lsort -integer -index 1 -decreasing [savestate::list_savestates_raw]] {
- set pres_str [lindex $i 0]
- lappend presentation $pres_str
- }
- return $presentation
- }
- proc menu_create_load_state {} {
- set menu_def \
- { execute menu_loadstate_exec
- font-size 8
- border-size 2
- width 200
- xpos 100
- ypos 120
- on-open {osd create rectangle "preview" -x 225 -y 5 -w 90 -h 70 -rgba 0x30303080 -scaled true}
- on-close {osd destroy "preview"}
- on-select menu_loadstate_select
- on-deselect menu_loadstate_deselect
- header { text "Load State"
- font-size 10
- post-spacing 6 }}
- set items [list_savestates -t]
- lappend menu_def presentation [get_savestates_list_presentation_sorted]
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_create_save_state {} {
- set items [concat [list "create new"] [list_savestates -t]]
- set menu_def \
- { execute menu_savestate_exec
- font-size 8
- border-size 2
- width 200
- xpos 100
- ypos 120
- on-open {osd create rectangle "preview" -x 225 -y 5 -w 90 -h 70 -rgba 0x30303080 -scaled true}
- on-close {osd destroy "preview"}
- on-select menu_loadstate_select
- on-deselect menu_loadstate_deselect
- header { text "Save State"
- font-size 10
- post-spacing 6 }}
- lappend menu_def presentation [concat [list "create new"] [get_savestates_list_presentation_sorted]]
- return [prepare_menu_list $items 10 $menu_def]
- }
- proc menu_loadstate_select {item} {
- set png $::env(OPENMSX_USER_DATA)/../savestates/${item}.png
- catch {osd create rectangle "preview.image" -relx 0.05 -rely 0.05 -w 80 -h 60 -image $png}
- }
- proc menu_loadstate_deselect {item} {
- osd destroy "preview.image"
- }
- proc menu_loadstate_exec {item} {
- if {[catch {loadstate $item} errorText]} {
- osd::display_message $errorText error
- } else {
- menu_close_all
- }
- }
- proc menu_savestate_exec {item} {
- if {$item eq "create new"} {
- set item [menu_free_savestate_name]
- confirm_save_state $item "Yes"
- menu_close_all
- } else {
- confirm_action "Overwrite $item?" osd_menu::confirm_save_state $item
- }
- }
- proc confirm_save_state {item result} {
- menu_close_top
- if {$result eq "Yes"} {
- if {[catch {savestate $item} errorText]} {
- osd::display_message $errorText error
- } else {
- osd::display_message "State saved to $item!"
- menu_close_all
- }
- }
- }
- proc menu_free_savestate_name {} {
- set existing [list_savestates]
- set i 1
- while 1 {
- set name [format "[guess_title savestate] %04d" $i]
- if {$name ni $existing} {
- return $name
- }
- incr i
- }
- }
- proc confirm_action {text action item} {
- set items [list "No" "Yes"]
- set menu_def [list execute [list $action $item] \
- font-size 8 \
- border-size 2 \
- width 210 \
- xpos 100 \
- ypos 100 \
- header [list text $text \
- font-size 10 \
- post-spacing 6 ]]
- osd_menu::menu_create [osd_menu::prepare_menu_list $items [llength $items] $menu_def]
- }
- proc menu_loadreplay_exec {item} {
- if {[catch {reverse loadreplay $item} errorText]} {
- osd::display_message $errorText error
- } else {
- menu_close_all
- }
- }
- proc menu_loadscript_exec {item} {
- if {[catch {source $item} errorText]} {
- osd::display_message $errorText error
- }
- }
- proc create_slot_actions_to_put_stuff_in_slot {slot path listtype} {
- return [list actions [list A [list osd_menu::menu_select_$listtype $slot $path]]]
- }
- proc drop_handler { event } {
- variable mediaslot_info
- lassign $event type filename
- set category [openmsx_info file_type_category $filename]
- set isdir [file isdirectory $filename]
- if {$category eq "unknown" && $isdir} {
- set category "disk"
- }
- set filetext "file"
- if {$isdir} {
- set filetext "folder"
- }
- if {$category eq "disk" || $category eq "rom"} {
- set longmediaslotname [dict get $mediaslot_info $category longmediaslotname]
- set listtype [dict get $mediaslot_info $category listtype]
- set mediabasecommand [dict get $mediaslot_info $category mediabasecommand]
- set slots [lsort [info command ${mediabasecommand}?]]
- if {[llength $slots] == 0} {
- osd::display_message "Can't handle dropped $filetext $filename, no $longmediaslotname present." error
- } elseif {[llength $slots] > 1} {
- set path $filename
- set menutitle "Select ${longmediaslotname}"
- set create_action_proc "create_slot_actions_to_put_stuff_in_slot"
- osd_menu::do_menu_open [create_slot_menu_def $slots $path $listtype $menutitle $create_action_proc]
- } else {
- osd_menu::menu_select_$listtype "${mediabasecommand}a" $filename true
- }
- } elseif {$category eq "laserdisc"} {
- if {[info command laserdiscplayer] ne ""} {; # only exists on some Pioneers
- osd_menu::menu_select_ld $filename
- } else {
- osd::display_message "Can't handle dropped $filetext $filename, no laser disc player present." error
- }
- } elseif {$category eq "cassette"} {
- if {[catch "machine_info connector cassetteport"]} {; # example: turboR
- osd::display_message "Can't handle dropped $filetext $filename, no cassette port present." error
- } else {
- osd_menu::menu_select_tape $filename
- }
- } elseif {$category eq "savestate"} {
- osd_menu::menu_loadstate_exec [file rootname $filename]
- } elseif {$category eq "replay"} {
- osd_menu::menu_loadreplay_exec $filename
- } elseif {$category eq "script"} {
- osd_menu::menu_loadscript_exec $filename
- } else {
- # stuff we can implement outside openMSX
- if {[file extension $filename] eq ".txt"} {
- type_from_file $filename
- } else {
- osd::display_message "Don't know how to handle dropped $filetext $filename..." error
- }
- }
- }
- # Keep openmsx console from interfering with the osd menu:
- # when the console is activated while the osd menu is already open, we want
- # to prevent the osd menu from receiving the keys that are pressed in the
- # console.
- variable old_console $::console
- proc console_input_layer {name1 name1 op} {
- global console
- variable old_console
- if {$console == $old_console} return
- set old_console $console
- if {$console} {
- activate_input_layer console -blocking
- } else {
- deactivate_input_layer console
- }
- }
- trace add variable ::console write [namespace code console_input_layer]
- namespace export main_menu_open
- namespace export main_menu_close
- namespace export main_menu_toggle
- } ;# namespace osd_menu
- namespace import osd_menu::*
|