_osd_menu.tcl 63 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038
  1. namespace eval osd_menu {
  2. set_help_text main_menu_open "Show the OSD menu."
  3. set_help_text main_menu_close "Remove the OSD menu."
  4. set_help_text main_menu_toggle "Toggle the OSD menu."
  5. variable is_dingux [string match dingux "[openmsx_info platform]"]
  6. variable scaling_available [expr {[lindex [lindex [openmsx_info setting scale_factor] 2] 1] > 1}]
  7. proc get_optional {dict_name key default} {
  8. upvar $dict_name d
  9. expr {[dict exists $d $key] ? [dict get $d $key] : $default}
  10. }
  11. proc set_optional {dict_name key value} {
  12. upvar $dict_name d
  13. if {![dict exists $d $key]} {
  14. dict set d $key $value
  15. }
  16. }
  17. variable menulevels 0
  18. # some variables for typing-in-menus support
  19. variable input_buffer ""
  20. variable input_last_time [openmsx_info realtime]
  21. variable input_timeout 1; #sec
  22. proc push_menu_info {} {
  23. variable menulevels
  24. incr menulevels 1
  25. set levelname "menuinfo_$menulevels"
  26. variable $levelname
  27. set $levelname [uplevel {dict create \
  28. name $name lst $lst menu_len $menu_len presentation $presentation \
  29. menutexts $menutexts selectinfo $selectinfo selectidx $selectidx \
  30. scrollidx $scrollidx on_close $on_close}]
  31. }
  32. proc peek_menu_info {} {
  33. variable menulevels
  34. uplevel upvar #0 osd_menu::menuinfo_$menulevels menuinfo
  35. }
  36. proc set_selectidx {value} {
  37. peek_menu_info
  38. dict set menuinfo selectidx $value
  39. }
  40. proc set_scrollidx {value} {
  41. peek_menu_info
  42. dict set menuinfo scrollidx $value
  43. }
  44. proc menu_create {menudef} {
  45. variable menulevels
  46. variable default_bg_color
  47. variable default_text_color
  48. variable default_select_color
  49. variable default_header_text_color
  50. set name "menu[expr {$menulevels + 1}]"
  51. set defactions [get_optional menudef "actions" ""]
  52. set bgcolor [get_optional menudef "bg-color" $default_bg_color]
  53. set deftextcolor [get_optional menudef "text-color" $default_text_color]
  54. set selectcolor [get_optional menudef "select-color" $default_select_color]
  55. set deffontsize [get_optional menudef "font-size" 12]
  56. set deffont [get_optional menudef "font" "skins/Vera.ttf.gz"]
  57. set bordersize [get_optional menudef "border-size" 0]
  58. set on_open [get_optional menudef "on-open" ""]
  59. set on_close [get_optional menudef "on-close" ""]
  60. osd create rectangle $name -scaled true -rgba $bgcolor -clip true \
  61. -borderrgba 0x000000ff -bordersize 0.5
  62. set y $bordersize
  63. set selectinfo [list]
  64. set menutexts [list]
  65. foreach itemdef [dict get $menudef items] {
  66. set selectable [get_optional itemdef "selectable" true]
  67. incr y [get_optional itemdef "pre-spacing" 0]
  68. set fontsize [get_optional itemdef "font-size" $deffontsize]
  69. set font [get_optional itemdef "font" $deffont]
  70. set textcolor [expr {$selectable
  71. ? [get_optional itemdef "text-color" $deftextcolor]
  72. : [get_optional itemdef "text-color" $default_header_text_color]}]
  73. set actions [get_optional itemdef "actions" ""]
  74. set on_select [get_optional itemdef "on-select" ""]
  75. set on_deselect [get_optional itemdef "on-deselect" ""]
  76. set textid "${name}.item${y}"
  77. set text [dict get $itemdef text]
  78. lappend menutexts $textid $text
  79. osd create text $textid -font $font -size $fontsize \
  80. -rgba $textcolor -x $bordersize -y $y
  81. if {$selectable} {
  82. set allactions [concat $defactions $actions]
  83. lappend selectinfo [list $y $fontsize $allactions $on_select $on_deselect]
  84. }
  85. incr y $fontsize
  86. incr y [get_optional itemdef "post-spacing" 0]
  87. }
  88. set width [dict get $menudef width]
  89. set height [expr {$y + $bordersize}]
  90. set xpos [get_optional menudef "xpos" [expr {(320 - $width) / 2}]]
  91. set ypos [get_optional menudef "ypos" [expr {(240 - $height) / 2}]]
  92. osd configure $name -x $xpos -y $ypos -w $width -h $height
  93. set selw [expr {$width - 2 * $bordersize}]
  94. osd create rectangle "${name}.selection" -z -1 -rgba $selectcolor \
  95. -x $bordersize -w $selw
  96. set lst [get_optional menudef "lst" ""]
  97. set menu_len [get_optional menudef "menu_len" 0]
  98. if {[llength $lst] > $menu_len} {
  99. set startheight 0
  100. if {[llength $selectinfo] > 0} {
  101. # there are selectable items. Start the scrollbar
  102. # at the top of the first selectable item
  103. # (skipping headers and stuff)
  104. set startheight [lindex $selectinfo 0 0]
  105. }
  106. osd create rectangle "${name}.scrollbar" -z -1 -rgba 0x00000010 \
  107. -relx 1.0 -x -6 -w 6 -relh 1.0 -h -$startheight -y $startheight -borderrgba 0x00000070 -bordersize 0.5
  108. osd create rectangle "${name}.scrollbar.thumb" -z -1 -rgba $default_select_color \
  109. -relw 1.0 -w -2 -x 1
  110. }
  111. set presentation [get_optional menudef "presentation" ""]
  112. set selectidx 0
  113. set scrollidx 0
  114. push_menu_info
  115. uplevel #0 $on_open
  116. menu_on_select $selectinfo $selectidx
  117. menu_refresh_top
  118. menu_update_scrollbar
  119. }
  120. proc menu_update_scrollbar {} {
  121. peek_menu_info
  122. set name [dict get $menuinfo name]
  123. if {[osd exists ${name}.scrollbar]} {
  124. set menu_len [dict get $menuinfo menu_len]
  125. set scrollidx [dict get $menuinfo scrollidx]
  126. set selectidx [dict get $menuinfo selectidx]
  127. set totalitems [llength [dict get $menuinfo lst]]
  128. set height [expr {1.0*$menu_len/$totalitems}]
  129. set minheight 0.05 ;# TODO: derive from width of bar
  130. set height [expr {$height > $minheight ? $height : $minheight}]
  131. set pos [expr {1.0*($scrollidx+$selectidx)/($totalitems-1)}]
  132. # scale the pos to the usable range
  133. set pos [expr {$pos*(1.0-$height)}]
  134. osd configure "${name}.scrollbar.thumb" -relh $height -rely $pos
  135. }
  136. }
  137. proc menu_refresh_top {} {
  138. peek_menu_info
  139. foreach {osdid text} [dict get $menuinfo menutexts] {
  140. set cmd [list subst $text]
  141. osd configure $osdid -text [uplevel #0 $cmd]
  142. }
  143. set selectinfo [dict get $menuinfo selectinfo]
  144. if {[llength $selectinfo] == 0} return
  145. set selectidx [dict get $menuinfo selectidx ]
  146. lassign [lindex $selectinfo $selectidx] sely selh
  147. osd configure "[dict get $menuinfo name].selection" -y $sely -h $selh
  148. }
  149. proc menu_close_top {} {
  150. variable menulevels
  151. peek_menu_info
  152. menu_on_deselect [dict get $menuinfo selectinfo] [dict get $menuinfo selectidx]
  153. uplevel #0 [dict get $menuinfo on_close]
  154. osd destroy [dict get $menuinfo name]
  155. unset menuinfo
  156. incr menulevels -1
  157. if {$menulevels == 0} {
  158. menu_last_closed
  159. }
  160. }
  161. proc menu_close_all {} {
  162. variable menulevels
  163. while {$menulevels} {
  164. menu_close_top
  165. }
  166. }
  167. proc menu_setting {cmd_result} {
  168. menu_refresh_top
  169. }
  170. proc menu_updown {delta} {
  171. peek_menu_info
  172. set num [llength [dict get $menuinfo selectinfo]]
  173. if {$num == 0} return
  174. set old_idx [dict get $menuinfo selectidx]
  175. menu_reselect [expr {($old_idx + $delta) % $num}]
  176. }
  177. proc menu_reselect {new_idx} {
  178. peek_menu_info
  179. set selectinfo [dict get $menuinfo selectinfo]
  180. set old_idx [dict get $menuinfo selectidx]
  181. menu_on_deselect $selectinfo $old_idx
  182. set_selectidx $new_idx
  183. menu_on_select $selectinfo $new_idx
  184. menu_refresh_top
  185. }
  186. proc menu_on_select {selectinfo selectidx} {
  187. set on_select [lindex $selectinfo $selectidx 3]
  188. uplevel #0 $on_select
  189. }
  190. proc menu_on_deselect {selectinfo selectidx} {
  191. set on_deselect [lindex $selectinfo $selectidx 4]
  192. uplevel #0 $on_deselect
  193. }
  194. proc menu_action {button} {
  195. # for any menu action, clear the input buffer
  196. variable input_buffer
  197. set input_buffer ""
  198. peek_menu_info
  199. set selectidx [dict get $menuinfo selectidx ]
  200. menu_action_idx $selectidx $button
  201. }
  202. proc menu_action_idx {idx button} {
  203. peek_menu_info
  204. set selectinfo [dict get $menuinfo selectinfo]
  205. set actions [lindex $selectinfo $idx 2]
  206. set_optional actions UP {osd_menu::menu_updown -1}
  207. set_optional actions DOWN {osd_menu::menu_updown 1}
  208. set_optional actions B {osd_menu::menu_close_top}
  209. set cmd [get_optional actions $button ""]
  210. uplevel #0 $cmd
  211. }
  212. proc get_mouse_coords {} {
  213. peek_menu_info
  214. set x 2; set y 2
  215. catch {
  216. set name [dict get $menuinfo name]
  217. lassign [osd info $name -mousecoord] x y
  218. }
  219. list $x $y
  220. }
  221. proc menu_get_mouse_idx {xy} {
  222. lassign $xy x y
  223. if {$x < 0 || 1 < $x || $y < 0 || 1 < $y} {return -1}
  224. peek_menu_info
  225. set name [dict get $menuinfo name]
  226. set yy [expr {$y * [osd info $name -h]}]
  227. set sel 0
  228. foreach i [dict get $menuinfo selectinfo] {
  229. lassign $i y h actions
  230. if {($y <= $yy) && ($yy < ($y + $h))} {
  231. return $sel
  232. }
  233. incr sel
  234. }
  235. return -1
  236. }
  237. proc menu_mouse_down {} {
  238. variable mouse_coord
  239. variable mouse_idx
  240. set mouse_coord [get_mouse_coords]
  241. set mouse_idx [menu_get_mouse_idx $mouse_coord]
  242. if {$mouse_idx != -1} {
  243. menu_reselect $mouse_idx
  244. }
  245. }
  246. proc menu_mouse_up {} {
  247. variable mouse_coord
  248. variable mouse_idx
  249. set mouse_coord [get_mouse_coords]
  250. set mouse_idx [menu_get_mouse_idx $mouse_coord]
  251. if {$mouse_idx != -1} {
  252. menu_action_idx $mouse_idx A
  253. }
  254. unset mouse_coord
  255. unset mouse_idx
  256. }
  257. proc menu_mouse_wheel {event} {
  258. lassign $event type1 type2 x y
  259. if {$y > 0} {
  260. osd_menu::menu_action UP
  261. } elseif {$y < 0} {
  262. osd_menu::menu_action DOWN
  263. }
  264. }
  265. proc menu_mouse_motion {} {
  266. variable mouse_coord
  267. variable mouse_idx
  268. if {![info exists mouse_coord]} return
  269. set new_mouse_coord [get_mouse_coords]
  270. set new_idx [menu_get_mouse_idx $new_mouse_coord]
  271. if {$new_idx != -1 && $new_idx != $mouse_idx} {
  272. menu_reselect $new_idx
  273. set mouse_coord $new_mouse_coord
  274. set mouse_idx $new_idx
  275. return
  276. }
  277. if {$mouse_idx != -1} {
  278. lassign $mouse_coord old_x old_y
  279. lassign $new_mouse_coord new_x new_y
  280. set delta_x [expr {$new_x - $old_x}]
  281. set delta_y [expr {$new_y - $old_y}]
  282. if {$delta_y > 0.1} {
  283. menu_action_idx $mouse_idx DOWN
  284. set mouse_coord $new_mouse_coord
  285. return
  286. } elseif {$delta_y < -0.1} {
  287. menu_action_idx $mouse_idx UP
  288. set mouse_coord $new_mouse_coord
  289. return
  290. } elseif {$delta_x > 0.1} {
  291. menu_action_idx $mouse_idx RIGHT
  292. set mouse_coord $new_mouse_coord
  293. return
  294. } elseif {$delta_x < -0.1} {
  295. menu_action_idx $mouse_idx LEFT
  296. set mouse_coord $new_mouse_coord
  297. return
  298. }
  299. }
  300. }
  301. user_setting create string osd_rom_path "OSD Rom Load Menu Last Known Path" $env(HOME)
  302. user_setting create string osd_disk_path "OSD Disk Load Menu Last Known Path" $env(HOME)
  303. user_setting create string osd_tape_path "OSD Tape Load Menu Last Known Path" $env(HOME)
  304. user_setting create string osd_hdd_path "OSD HDD Load Menu Last Known Path" $env(HOME)
  305. user_setting create string osd_ld_path "OSD LD Load Menu Last Known Path" $env(HOME)
  306. if {![file exists $::osd_rom_path] || ![file readable $::osd_rom_path]} {
  307. # revert to default (should always exist)
  308. unset ::osd_rom_path
  309. }
  310. if {![file exists $::osd_disk_path] || ![file readable $::osd_disk_path]} {
  311. # revert to default (should always exist)
  312. unset ::osd_disk_path
  313. }
  314. if {![file exists $::osd_tape_path] || ![file readable $::osd_tape_path]} {
  315. # revert to default (should always exist)
  316. unset ::osd_tape_path
  317. }
  318. if {![file exists $::osd_hdd_path] || ![file readable $::osd_hdd_path]} {
  319. # revert to default (should always exist)
  320. unset ::osd_hdd_path
  321. }
  322. if {![file exists $::osd_ld_path] || ![file readable $::osd_ld_path]} {
  323. # revert to default (should always exist)
  324. unset ::osd_ld_path
  325. }
  326. variable taperecordings_directory [file normalize $::env(OPENMSX_USER_DATA)/../taperecordings]
  327. proc main_menu_open {} {
  328. do_menu_open [create_main_menu]
  329. }
  330. proc do_menu_open {top_menu} {
  331. variable is_dingux
  332. # close console, because the menu interferes with it
  333. set ::console off
  334. # also remove other OSD controlled widgets (like the osd keyboard)
  335. if {[info exists ::osd_control::close]} {
  336. eval $::osd_control::close
  337. }
  338. # end tell how to close this widget
  339. namespace eval ::osd_control {set close ::osd_menu::main_menu_close}
  340. menu_create $top_menu
  341. set ::pause true
  342. # TODO make these bindings easier to customize
  343. bind -layer osd_menu "OSDcontrol UP PRESS" -repeat {osd_menu::menu_action UP }
  344. bind -layer osd_menu "OSDcontrol DOWN PRESS" -repeat {osd_menu::menu_action DOWN }
  345. bind -layer osd_menu "OSDcontrol LEFT PRESS" -repeat {osd_menu::menu_action LEFT }
  346. bind -layer osd_menu "OSDcontrol RIGHT PRESS" -repeat {osd_menu::menu_action RIGHT}
  347. bind -layer osd_menu "mouse button1 down" {osd_menu::menu_mouse_down}
  348. bind -layer osd_menu "mouse button1 up" {osd_menu::menu_mouse_up}
  349. bind -layer osd_menu "mouse button3 up" {osd_menu::menu_close_top}
  350. bind -layer osd_menu "mouse motion" {osd_menu::menu_mouse_motion}
  351. bind -layer osd_menu "mouse wheel" -event {osd_menu::menu_mouse_wheel}
  352. if {$is_dingux} {
  353. bind -layer osd_menu "keyb LCTRL" {osd_menu::menu_action A }
  354. bind -layer osd_menu "keyb LALT" {osd_menu::menu_action B }
  355. } else {
  356. bind -layer osd_menu "OSDcontrol A PRESS" {osd_menu::menu_action A }
  357. bind -layer osd_menu "OSDcontrol B PRESS" {osd_menu::menu_action B }
  358. # on Android, use BACK button to go back in menus
  359. bind -layer osd_menu "keyb BACK" {osd_menu::menu_action B }
  360. }
  361. bind -layer osd_menu "CTRL+UP" {osd_menu::select_menu_idx 0}
  362. bind -layer osd_menu "CTRL+LEFT" {osd_menu::select_menu_idx 0}
  363. bind -layer osd_menu "keyb HOME" {osd_menu::select_menu_idx 0}
  364. 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}
  365. foreach char $alphanum {
  366. bind -layer osd_menu "keyb $char" "osd_menu::handle_keyboard_input $char"
  367. }
  368. activate_input_layer osd_menu -blocking
  369. }
  370. proc main_menu_close {} {
  371. menu_close_all
  372. }
  373. proc main_menu_toggle {} {
  374. variable menulevels
  375. if {$menulevels} {
  376. # there is at least one menu open, close it
  377. menu_close_all
  378. } else {
  379. # none open yet, open main menu
  380. main_menu_open
  381. }
  382. }
  383. proc menu_last_closed {} {
  384. set ::pause false
  385. after realtime 0 {deactivate_input_layer osd_menu}
  386. namespace eval ::osd_control {unset close}
  387. }
  388. proc prepare_menu_list {lst num menudef} {
  389. set execute [dict get $menudef execute]
  390. set header [dict get $menudef header]
  391. set item_extra [get_optional menudef item ""]
  392. set on_select [get_optional menudef on-select ""]
  393. set on_deselect [get_optional menudef on-deselect ""]
  394. set presentation [get_optional menudef presentation $lst]
  395. # 'assert': presentation should have same length as item list!
  396. if {[llength $presentation] != [llength $lst]} {
  397. error "Presentation should be of same length as item list!"
  398. }
  399. dict set menudef presentation $presentation
  400. lappend header "selectable" "false"
  401. set items [list $header]
  402. set lst_len [llength $lst]
  403. set menu_len [expr {$lst_len < $num ? $lst_len : $num}]
  404. for {set i 0} {$i < $menu_len} {incr i} {
  405. set actions [list "A" "osd_menu::list_menu_item_exec {$execute} $i"]
  406. if {$i == 0} {
  407. lappend actions "UP" "osd_menu::move_selection -1"
  408. }
  409. if {$i == ($menu_len - 1)} {
  410. lappend actions "DOWN" "osd_menu::move_selection 1"
  411. }
  412. lappend actions "LEFT" "osd_menu::move_selection -$menu_len"
  413. lappend actions "RIGHT" "osd_menu::move_selection $menu_len"
  414. set item [list "text" "\[osd_menu::list_menu_item_show $i\]" \
  415. "actions" $actions]
  416. if {$on_select ne ""} {
  417. lappend item "on-select" "osd_menu::list_menu_item_select $i $on_select"
  418. }
  419. if {$on_deselect ne ""} {
  420. lappend item "on-deselect" "osd_menu::list_menu_item_select $i $on_deselect"
  421. }
  422. lappend items [concat $item $item_extra]
  423. }
  424. dict set menudef items $items
  425. dict set menudef lst $lst
  426. dict set menudef menu_len $menu_len
  427. return $menudef
  428. }
  429. proc list_menu_item_exec {execute pos} {
  430. peek_menu_info
  431. {*}$execute [lindex [dict get $menuinfo lst] [expr {$pos + [dict get $menuinfo scrollidx]}]]
  432. }
  433. proc list_menu_item_show {pos} {
  434. peek_menu_info
  435. return [lindex [dict get $menuinfo presentation] [expr {$pos + [dict get $menuinfo scrollidx]}]]
  436. }
  437. proc list_menu_item_select {pos select_proc} {
  438. peek_menu_info
  439. $select_proc [lindex [dict get $menuinfo lst] [expr {$pos + [dict get $menuinfo scrollidx]}]]
  440. }
  441. proc move_selection {delta} {
  442. peek_menu_info
  443. set lst_last [expr {[llength [dict get $menuinfo lst]] - 1}]
  444. set scrollidx [dict get $menuinfo scrollidx]
  445. set selectidx [dict get $menuinfo selectidx]
  446. set old_itemidx [expr {$scrollidx + $selectidx}]
  447. set new_itemidx [expr {$old_itemidx + $delta}]
  448. if {$new_itemidx < 0} {
  449. # Before first element
  450. if {$old_itemidx == 0} {
  451. # if first element was already selected, wrap to last
  452. set new_itemidx $lst_last
  453. } else {
  454. # otherwise, clamp to first element
  455. set new_itemidx 0
  456. }
  457. } elseif {$new_itemidx > $lst_last} {
  458. # After last element
  459. if {$old_itemidx == $lst_last} {
  460. # if last element was already selected, wrap to first
  461. set new_itemidx 0
  462. } else {
  463. # otherwise clam to last element
  464. set new_itemidx $lst_last
  465. }
  466. }
  467. select_menu_idx $new_itemidx
  468. }
  469. proc select_menu_idx {itemidx} {
  470. peek_menu_info
  471. set menu_len [dict get $menuinfo menu_len]
  472. set scrollidx [dict get $menuinfo scrollidx]
  473. set selectidx [dict get $menuinfo selectidx]
  474. set selectinfo [dict get $menuinfo selectinfo]
  475. menu_on_deselect $selectinfo $selectidx
  476. set selectidx [expr {$itemidx - $scrollidx}]
  477. if {$selectidx < 0} {
  478. incr scrollidx $selectidx
  479. set selectidx 0
  480. } elseif {($menu_len > 0) && ($selectidx >= $menu_len)} {
  481. set selectidx [expr {$menu_len - 1}]
  482. set scrollidx [expr {$itemidx - $selectidx}]
  483. }
  484. set_selectidx $selectidx
  485. set_scrollidx $scrollidx
  486. menu_on_select $selectinfo $selectidx
  487. menu_refresh_top
  488. menu_update_scrollbar
  489. }
  490. proc select_menu_item {item} {
  491. peek_menu_info
  492. set index [lsearch -exact [dict get $menuinfo lst] $item]
  493. if {$index == -1} return
  494. select_menu_idx $index
  495. }
  496. proc handle_keyboard_input {char} {
  497. variable input_buffer
  498. variable input_last_time
  499. variable input_timeout
  500. set current_time [openmsx_info realtime]
  501. if {[expr {$current_time - $input_last_time}] < $input_timeout} {
  502. set input_buffer "$input_buffer$char"
  503. } else {
  504. set input_buffer $char
  505. }
  506. set input_last_time $current_time
  507. osd_menu::select_next_menu_item_starting_with $input_buffer
  508. }
  509. proc select_next_menu_item_starting_with {text} {
  510. peek_menu_info
  511. set items [dict get $menuinfo presentation]
  512. if {[llength $items] == 0} return
  513. set selectidx [dict get $menuinfo selectidx]
  514. set scrollidx [dict get $menuinfo scrollidx]
  515. set itemidx [expr {$scrollidx + $selectidx}]
  516. # start after the current item if this is a new search
  517. if {[string length $text] == 1} {
  518. incr itemidx
  519. }
  520. # use the list twice to wrap
  521. set index [lsearch -glob -nocase -start $itemidx [concat $items $items] "$text*"]
  522. if {$index == -1} return
  523. set index [expr {$index % [llength $items]}]
  524. select_menu_idx $index
  525. }
  526. variable mediaslot_info [dict create \
  527. "rom" [dict create mediabasecommand "cart" mediapath "::osd_rom_path" listtype "rom" itemtext "Load ROM" shortmediaslotname "slot" longmediaslotname "cartridge slot"]\
  528. "disk" [dict create mediabasecommand "disk" mediapath "::osd_disk_path" listtype "disk" itemtext "Insert disk" shortmediaslotname "drive" longmediaslotname "disk drive"]\
  529. #"cassette" [dict create mediabasecommand "cassetteplayer" mediapath "::osd_tape_path" listtype "tape" itemtext "Set tape" shortmediaslotname "xxx" longmediaslotname "cassette player"]\
  530. ]
  531. proc create_slot_actions_to_select_file {slot path listtype} {
  532. # 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.
  533. 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\]}"]]
  534. }
  535. proc get_slot_str {slot} {
  536. return [string toupper [string index $slot end]]
  537. }
  538. proc create_slot_menu_def {slots path listtype menutitle create_action_proc} {
  539. set menudef {
  540. font-size 8
  541. border-size 2
  542. width 150
  543. xpos 100
  544. ypos 80
  545. }
  546. lappend items [list text $menutitle\
  547. font-size 12\
  548. post-spacing 6\
  549. selectable false\
  550. ]
  551. foreach slot $slots {
  552. set slotcontent "(empty)"
  553. if {![string match "empty*" [lindex [$slot] 2]]} {
  554. set slotcontent [file tail [lindex [$slot] 1]]
  555. }
  556. lappend items [list text "[get_slot_str $slot]: $slotcontent" {*}[$create_action_proc $slot $path $listtype]]
  557. }
  558. dict set menudef items $items
  559. return $menudef
  560. }
  561. proc create_media_menu_items {file_type_category} {
  562. variable mediaslot_info
  563. set longmediaslotname [dict get $mediaslot_info $file_type_category longmediaslotname]
  564. set shortmediaslotname [dict get $mediaslot_info $file_type_category shortmediaslotname]
  565. set mediapath [dict get $mediaslot_info $file_type_category mediapath]
  566. set listtype [dict get $mediaslot_info $file_type_category listtype]
  567. set itemtext [dict get $mediaslot_info $file_type_category itemtext]
  568. set mediabasecommand [dict get $mediaslot_info $file_type_category mediabasecommand]
  569. if {[catch ${mediabasecommand}a]} {; # example: Philips NMS 801 without cart slot, or no diskdrives. TODO: make more generic
  570. lappend menuitems [list text "(No $longmediaslotname present...)"\
  571. selectable false\
  572. text-color 0x808080ff\
  573. ]
  574. } else {
  575. set slots [lsort [info command ${mediabasecommand}?]]; # TODO: make more generic
  576. if {[llength $slots] <= 2} {
  577. foreach slot $slots {
  578. lappend menuitems [list text "${itemtext}... (${shortmediaslotname} [get_slot_str $slot])" {*}[create_slot_actions_to_select_file $slot $mediapath $listtype]]
  579. }
  580. } else {
  581. set slot [lindex $slots 0]
  582. lappend menuitems [list text "${itemtext}... (${shortmediaslotname} [get_slot_str $slot])" {*}[create_slot_actions_to_select_file $slot $mediapath $listtype]]
  583. 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\}\]"]]
  584. }
  585. }
  586. return $menuitems
  587. }
  588. #
  589. # definitions of menus
  590. #
  591. proc create_main_menu {} {
  592. set menu_def {
  593. font-size 10
  594. border-size 2
  595. width 160
  596. }
  597. lappend items { text "[openmsx_info version]"
  598. font-size 12
  599. post-spacing 6
  600. selectable false }
  601. lappend items {*}[create_media_menu_items "rom"]
  602. lappend items {*}[create_media_menu_items "disk"]
  603. if {[info command hda] ne ""} {; # only exists when hard disk extension available
  604. foreach drive [lrange [lsort [info command hd?]] 0 1] {
  605. set drive_str [string toupper [string index $drive end]]
  606. lappend items [list text "Change HD/SD image... (drive $drive_str)" \
  607. 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\]}"]]
  608. }
  609. }
  610. if {[info command laserdiscplayer] ne ""} {; # only exists on some Pioneers
  611. lappend items { text "Load LaserDisc..."
  612. 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]}} }
  613. }
  614. }
  615. if {[catch "machine_info connector cassetteport"]} {; # example: turboR
  616. lappend items { text "(No cassette port present...)"
  617. selectable false
  618. text-color 0x808080ff
  619. post-spacing 3
  620. }
  621. } else {
  622. lappend items { text "Set Tape..."
  623. 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]]}} }
  624. post-spacing 3 }
  625. }
  626. lappend items { text "Save State..."
  627. actions { A { osd_menu::menu_create [osd_menu::menu_create_save_state] }}}
  628. lappend items { text "Load State..."
  629. actions { A { osd_menu::menu_create [osd_menu::menu_create_load_state] }}
  630. post-spacing 3 }
  631. lappend items { text "Hardware..."
  632. actions { A { osd_menu::menu_create [osd_menu::create_hardware_menu] }}
  633. post-spacing 3 }
  634. lappend items { text "Misc Settings..."
  635. actions { A { osd_menu::menu_create $osd_menu::misc_setting_menu }}}
  636. lappend items { text "Sound Settings..."
  637. actions { A { osd_menu::menu_create $osd_menu::sound_setting_menu }}}
  638. lappend items { text "Video Settings..."
  639. actions { A { osd_menu::menu_create [osd_menu::create_video_setting_menu] }}
  640. post-spacing 3 }
  641. lappend items { text "Advanced..."
  642. actions { A { osd_menu::menu_create $osd_menu::advanced_menu }}
  643. post-spacing 10 }
  644. lappend items { text "Reset MSX"
  645. actions { A { reset; osd_menu::menu_close_all }}}
  646. lappend items { text "Exit openMSX"
  647. actions { A quitmenu::quit_menu }}
  648. dict set menu_def items $items
  649. return $menu_def
  650. }
  651. set misc_setting_menu {
  652. font-size 8
  653. border-size 2
  654. width 150
  655. xpos 100
  656. ypos 120
  657. items {{ text "Misc Settings"
  658. font-size 10
  659. post-spacing 6
  660. selectable false }
  661. { text "Speed: $speed"
  662. actions { LEFT { osd_menu::menu_setting [incr speed -1] }
  663. RIGHT { osd_menu::menu_setting [incr speed 1] }}}
  664. { text "Minimal Frameskip: $minframeskip"
  665. actions { LEFT { osd_menu::menu_setting [incr minframeskip -1] }
  666. RIGHT { osd_menu::menu_setting [incr minframeskip 1] }}}
  667. { text "Maximal Frameskip: $maxframeskip"
  668. actions { LEFT { osd_menu::menu_setting [incr maxframeskip -1] }
  669. RIGHT { osd_menu::menu_setting [incr maxframeskip 1] }}}}}
  670. set resampler_desc [dict create fast "fast (but low quality)" blip "blip (good speed/quality)" hq "hq (best but slow on Android)"]
  671. set sound_setting_menu {
  672. font-size 8
  673. border-size 2
  674. width 180
  675. xpos 100
  676. ypos 120
  677. items {{ text "Sound Settings"
  678. font-size 10
  679. post-spacing 6
  680. selectable false }
  681. { text "Volume: $master_volume"
  682. actions { LEFT { osd_menu::menu_setting [incr master_volume -5] }
  683. RIGHT { osd_menu::menu_setting [incr master_volume 5] }}}
  684. { text "Mute: $mute"
  685. actions { LEFT { osd_menu::menu_setting [cycle_back mute] }
  686. RIGHT { osd_menu::menu_setting [cycle mute] }}}
  687. { text "Individual Sound Device Settings..."
  688. actions { A { osd_menu::menu_create [osd_menu::menu_create_sound_device_list]}}}
  689. { text "Resampler: [osd_menu::get_resampler_presentation $resampler]"
  690. actions { LEFT { osd_menu::menu_setting [cycle_back resampler] }
  691. RIGHT { osd_menu::menu_setting [cycle resampler] }}}}}
  692. 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"]
  693. proc menu_create_sound_device_list {} {
  694. set menu_def {
  695. execute menu_sound_device_select_exec
  696. font-size 8
  697. border-size 2
  698. width 200
  699. xpos 110
  700. ypos 130
  701. header { text "Select Sound Chip"
  702. font-size 10
  703. post-spacing 6 }}
  704. set items [machine_info sounddevice]
  705. return [prepare_menu_list $items 5 $menu_def]
  706. }
  707. proc menu_sound_device_select_exec {item} {
  708. menu_create [create_sound_device_settings_menu $item]
  709. select_menu_item $item
  710. }
  711. proc create_sound_device_settings_menu {device} {
  712. set ypos 140
  713. set menu_def [list \
  714. font-size 8 \
  715. border-size 2 \
  716. width 210 \
  717. xpos 120 \
  718. ypos $ypos]
  719. lappend items [list text "$device Settings" \
  720. font-size 10 \
  721. post-spacing 6 \
  722. selectable false]
  723. # volume and balance
  724. foreach aspect [list volume balance] {
  725. set var_name ::${device}_${aspect}
  726. set item [list]
  727. lappend item "text"
  728. set first [string range $aspect 0 0]
  729. set rest [string range $aspect 1 end]
  730. set first [string toupper $first]
  731. set capped_aspect "${first}${rest}"
  732. lappend item "$capped_aspect: \[[list set $var_name]]"
  733. lappend item "actions"
  734. set actions [list]
  735. lappend actions "LEFT"
  736. lappend actions "osd_menu::menu_setting \[[list incr $var_name -5]]"
  737. lappend actions "RIGHT"
  738. lappend actions "osd_menu::menu_setting \[[list incr $var_name 5]]"
  739. lappend item $actions
  740. lappend items $item
  741. }
  742. # channel mute
  743. set channel_count [soundchip_utils::get_num_channels $device]
  744. for {set channel 1} {$channel <= $channel_count} {incr channel} {
  745. set chmute_var_name ${device}_ch${channel}_mute
  746. set item [list]
  747. lappend item "text"
  748. set pretext ""
  749. if {$channel_count > 1} {
  750. set pretext "Channel $channel "
  751. }
  752. lappend item "${pretext}Mute: \[[list set $chmute_var_name]]"
  753. lappend item "actions"
  754. set actions [list]
  755. lappend actions "LEFT"
  756. lappend actions "osd_menu::menu_setting \[[list cycle_back $chmute_var_name]]"
  757. lappend actions "RIGHT"
  758. lappend actions "osd_menu::menu_setting \[[list cycle $chmute_var_name]]"
  759. lappend item $actions
  760. lappend items $item
  761. }
  762. # adjust menu position for longer lists
  763. # TODO: make this less magic
  764. if {$channel_count > 8} {;# more won't fit
  765. dict set menu_def ypos [expr {$ypos - round(($channel_count - 8) * ($ypos - 10)/16)}]
  766. }
  767. dict set menu_def items $items
  768. return $menu_def
  769. }
  770. proc create_video_setting_menu {} {
  771. variable scaling_available
  772. set menu_def {
  773. font-size 8
  774. border-size 2
  775. width 210
  776. xpos 100
  777. ypos 110
  778. }
  779. lappend items { text "Video Settings"
  780. font-size 10
  781. post-spacing 6
  782. selectable false }
  783. if {[expr {[lindex [lindex [openmsx_info setting videosource] 2] 1] > 1}]} {
  784. lappend items { text "Video source: $videosource"
  785. actions { LEFT { osd_menu::menu_setting [cycle_back videosource] }
  786. RIGHT { osd_menu::menu_setting [cycle videosource] }}
  787. post-spacing 6}
  788. }
  789. if {$scaling_available} {
  790. lappend items { text "Scaler: $scale_algorithm"
  791. actions { LEFT { osd_menu::menu_setting [cycle_back scale_algorithm] }
  792. RIGHT { osd_menu::menu_setting [cycle scale_algorithm] }}}
  793. # only add scale factor setting if it can actually be changed
  794. set scale_minmax [lindex [openmsx_info setting scale_factor] 2]
  795. if {[expr {[lindex $scale_minmax 0] != [lindex $scale_minmax 1]}]} {
  796. lappend items { text "Scale Factor: ${scale_factor}x"
  797. actions { LEFT { osd_menu::menu_setting [incr scale_factor -1] }
  798. RIGHT { osd_menu::menu_setting [incr scale_factor 1] }}}
  799. }
  800. }
  801. lappend items { text "Horizontal Stretch: [osd_menu::get_horizontal_stretch_presentation $horizontal_stretch]"
  802. actions { A { osd_menu::menu_create [osd_menu::menu_create_stretch_list]; osd_menu::select_menu_item $horizontal_stretch }}
  803. post-spacing 6 }
  804. if {$scaling_available} {
  805. lappend items { text "Scanline: $scanline%"
  806. actions { LEFT { osd_menu::menu_setting [incr scanline -1] }
  807. RIGHT { osd_menu::menu_setting [incr scanline 1] }}}
  808. lappend items { text "Blur: $blur%"
  809. actions { LEFT { osd_menu::menu_setting [incr blur -1] }
  810. RIGHT { osd_menu::menu_setting [incr blur 1] }}}
  811. }
  812. if {$::renderer eq "SDLGL-PP"} {
  813. lappend items { text "Glow: $glow%"
  814. actions { LEFT { osd_menu::menu_setting [incr glow -1] }
  815. RIGHT { osd_menu::menu_setting [incr glow 1] }}}
  816. lappend items { text "Display Deform: $display_deform"
  817. actions { LEFT { osd_menu::menu_setting [cycle_back display_deform] }
  818. RIGHT { osd_menu::menu_setting [cycle display_deform] }}}
  819. }
  820. lappend items { text "Noise: $noise%"
  821. actions { LEFT { osd_menu::menu_setting [set noise [expr $noise - 1]] }
  822. RIGHT { osd_menu::menu_setting [set noise [expr $noise + 1]] }}
  823. post-spacing 6}
  824. lappend items { text "Enforce VDP Sprites-per-line Limit: $limitsprites"
  825. actions { LEFT { osd_menu::menu_setting [cycle_back limitsprites] }
  826. RIGHT { osd_menu::menu_setting [cycle limitsprites] }}}
  827. dict set menu_def items $items
  828. return $menu_def
  829. }
  830. proc create_hardware_menu {} {
  831. set menu_def {
  832. font-size 8
  833. border-size 2
  834. width 175
  835. xpos 100
  836. ypos 120
  837. }
  838. lappend items { text "Hardware"
  839. font-size 10
  840. post-spacing 6
  841. selectable false }
  842. lappend items { text "Change Machine..."
  843. actions { A { osd_menu::menu_create [osd_menu::menu_create_load_machine_list]; catch { osd_menu::select_menu_item [machine_info config_name]} }}}
  844. lappend items { text "Set Current Machine as Default"
  845. actions { A { set ::default_machine [machine_info config_name]; osd_menu::menu_close_top }}}
  846. lappend items { text "Extensions..."
  847. actions { A { osd_menu::menu_create $osd_menu::extensions_menu }}}
  848. lappend items { text "Connectors..."
  849. actions { A { osd_menu::menu_create [osd_menu::menu_create_connectors_list] }}}
  850. if {![catch {openmsx_info setting firmwareswitch}]} {
  851. lappend items { text "Firmware switch active: $::firmwareswitch"
  852. actions { LEFT { osd_menu::menu_setting [cycle_back firmwareswitch] }
  853. RIGHT { osd_menu::menu_setting [cycle firmwareswitch] }}}
  854. }
  855. dict set menu_def items $items
  856. return $menu_def
  857. }
  858. set extensions_menu {
  859. font-size 8
  860. border-size 2
  861. width 175
  862. xpos 100
  863. ypos 120
  864. items {{ text "Extensions"
  865. font-size 10
  866. post-spacing 6
  867. selectable false }
  868. { text "Add..."
  869. actions { A { osd_menu::menu_create [osd_menu::menu_create_extensions_list] }}}
  870. { text "Remove..."
  871. actions { A { osd_menu::menu_create [osd_menu::menu_create_plugged_extensions_list] }}}}}
  872. set advanced_menu {
  873. font-size 8
  874. border-size 2
  875. width 175
  876. xpos 100
  877. ypos 120
  878. items {{ text "Advanced"
  879. font-size 10
  880. post-spacing 6
  881. selectable false }
  882. { text "Manage Running Machines..."
  883. actions { A { osd_menu::menu_create $osd_menu::running_machines_menu }}}
  884. { text "Toys and Utilities..."
  885. actions { A { osd_menu::menu_create [osd_menu::menu_create_toys_list] }}}}}
  886. set running_machines_menu {
  887. font-size 8
  888. border-size 2
  889. width 175
  890. xpos 100
  891. ypos 120
  892. items {{ text "Manage Running Machines"
  893. font-size 10
  894. post-spacing 6
  895. selectable false }
  896. { text "Select Running Machine Tab: [utils::get_machine_display_name]"
  897. actions { A { osd_menu::menu_create [osd_menu::menu_create_running_machine_list] }}}
  898. { text "New Running Machine Tab"
  899. actions { A { osd_menu::menu_create [osd_menu::menu_create_load_machine_list "add"] }}}
  900. { text "Close Current Machine Tab"
  901. actions { A { set old_active_machine [activate_machine]; cycle_machine; delete_machine $old_active_machine }}}}}
  902. proc menu_create_running_machine_list {} {
  903. set menu_def {
  904. execute menu_machine_tab_select_exec
  905. font-size 8
  906. border-size 2
  907. width 200
  908. xpos 110
  909. ypos 130
  910. header { text "Select Running Machine"
  911. font-size 10
  912. post-spacing 6 }}
  913. set items [utils::get_ordered_machine_list]
  914. set presentation [list]
  915. foreach i $items {
  916. if {[activate_machine] eq $i} {
  917. set postfix_text "current"
  918. } else {
  919. set postfix_text [utils::get_machine_time $i]
  920. }
  921. lappend presentation [format "%s (%s)" [utils::get_machine_display_name ${i}] $postfix_text]
  922. }
  923. lappend menu_def presentation $presentation
  924. return [prepare_menu_list $items 5 $menu_def]
  925. }
  926. proc menu_machine_tab_select_exec {item} {
  927. menu_close_top
  928. activate_machine $item
  929. }
  930. proc get_resampler_presentation { value } {
  931. if {[dict exists $osd_menu::resampler_desc $value]} {
  932. return [dict get $osd_menu::resampler_desc $value]
  933. } else {
  934. return $value
  935. }
  936. }
  937. proc get_horizontal_stretch_presentation { value } {
  938. if {[dict exists $osd_menu::horizontal_stretch_desc $value]} {
  939. return [dict get $osd_menu::horizontal_stretch_desc $value]
  940. } else {
  941. return "custom: $::horizontal_stretch"
  942. }
  943. }
  944. proc menu_create_stretch_list {} {
  945. set menu_def [list \
  946. execute menu_stretch_exec \
  947. font-size 8 \
  948. border-size 2 \
  949. width 150 \
  950. xpos 110 \
  951. ypos 130 \
  952. header { text "Select Horizontal Stretch:"
  953. font-size 10
  954. post-spacing 6 }]
  955. set items [list]
  956. set presentation [list]
  957. set values [dict keys $osd_menu::horizontal_stretch_desc]
  958. if {$::horizontal_stretch ni $values} {
  959. lappend values $::horizontal_stretch
  960. }
  961. foreach value $values {
  962. lappend items $value
  963. lappend presentation [osd_menu::get_horizontal_stretch_presentation $value]
  964. }
  965. lappend menu_def presentation $presentation
  966. return [prepare_menu_list $items 6 $menu_def]
  967. }
  968. proc menu_stretch_exec {value} {
  969. set ::horizontal_stretch $value
  970. menu_close_top
  971. menu_refresh_top
  972. }
  973. # Returns list of machines/extensions, but try to filter out duplicates caused
  974. # by symlinks (e.g. turbor.xml -> Panasonic_FS-A1GT.xml). What this does not
  975. # catch is a symlink in the systemdir (so link also pointing to the systemdir)
  976. # and a similarly named file in the userdir. This situation does occur on my
  977. # development setup, but it shouldn't happen for regular users.
  978. proc get_filtered_configs {type} {
  979. set result [list]
  980. set configs [list]
  981. foreach t [openmsx_info $type] {
  982. # try both <name>.xml and <name>/hardwareconfig.xml
  983. set conf [data_file $type/$t.xml]
  984. if {![file exists $conf]} {
  985. set conf [data_file $type/$t/hardwareconfig.xml]
  986. }
  987. # follow symlink (on platforms that support links)
  988. catch {
  989. set conf [file join [file dirname $conf] [file readlink $conf]]
  990. }
  991. # only add if the (possibly resolved link) hasn't been seen before
  992. if {$conf ni $configs} {
  993. lappend configs $conf
  994. lappend result $t
  995. }
  996. }
  997. return $result
  998. }
  999. proc menu_create_load_machine_list {{mode "replace"}} {
  1000. if {$mode eq "replace"} {
  1001. set proc_to_exec osd_menu::menu_load_machine_exec_replace
  1002. } elseif {$mode eq "add"} {
  1003. set proc_to_exec osd_menu::menu_load_machine_exec_add
  1004. } else {
  1005. error "Undefined mode: $mode"
  1006. }
  1007. set menu_def [list \
  1008. execute $proc_to_exec \
  1009. font-size 8 \
  1010. border-size 2 \
  1011. width 200 \
  1012. xpos 110 \
  1013. ypos 130 \
  1014. header { text "Select Machine to Run"
  1015. font-size 10
  1016. post-spacing 6 }]
  1017. set items [get_filtered_configs machines]
  1018. foreach i $items {
  1019. set extra_info ""
  1020. if {$i eq $::default_machine} {
  1021. set extra_info " (default)"
  1022. }
  1023. lappend presentation "[utils::get_machine_display_name_by_config_name $i]$extra_info"
  1024. }
  1025. set items_sorted [list]
  1026. set presentation_sorted [list]
  1027. foreach i [lsort -dictionary -indices $presentation] {
  1028. lappend presentation_sorted [lindex $presentation $i]
  1029. lappend items_sorted [lindex $items $i]
  1030. }
  1031. lappend menu_def presentation $presentation_sorted
  1032. return [prepare_menu_list $items_sorted 10 $menu_def]
  1033. }
  1034. proc menu_load_machine_exec_replace {item} {
  1035. if {[catch {machine $item} errorText]} {
  1036. osd::display_message $errorText error
  1037. } else {
  1038. menu_close_all
  1039. }
  1040. }
  1041. proc menu_load_machine_exec_add {item} {
  1042. set id [create_machine]
  1043. set err [catch {${id}::load_machine $item} error_result]
  1044. if {$err} {
  1045. delete_machine $id
  1046. osd::display_message "Error starting [utils::get_machine_display_name_by_config_name $item]: $error_result" error
  1047. } else {
  1048. menu_close_top
  1049. activate_machine $id
  1050. }
  1051. }
  1052. proc menu_create_extensions_list {} {
  1053. set menu_def {
  1054. execute menu_add_extension_exec
  1055. font-size 8
  1056. border-size 2
  1057. width 200
  1058. xpos 110
  1059. ypos 130
  1060. header { text "Select Extension to Add"
  1061. font-size 10
  1062. post-spacing 6 }}
  1063. set items [get_filtered_configs extensions]
  1064. set presentation [list]
  1065. foreach i $items {
  1066. lappend presentation [utils::get_extension_display_name_by_config_name $i]
  1067. }
  1068. set items_sorted [list]
  1069. set presentation_sorted [list]
  1070. foreach i [lsort -dictionary -indices $presentation] {
  1071. lappend presentation_sorted [lindex $presentation $i]
  1072. lappend items_sorted [lindex $items $i]
  1073. }
  1074. lappend menu_def presentation $presentation_sorted
  1075. return [prepare_menu_list $items_sorted 10 $menu_def]
  1076. }
  1077. proc menu_add_extension_exec {item} {
  1078. if {[catch {ext $item} errorText]} {
  1079. osd::display_message $errorText error
  1080. } else {
  1081. menu_close_all
  1082. }
  1083. }
  1084. proc menu_create_plugged_extensions_list {} {
  1085. set menu_def {
  1086. execute menu_remove_extension_exec
  1087. font-size 8
  1088. border-size 2
  1089. width 200
  1090. xpos 110
  1091. ypos 130
  1092. header { text "Select Extension to Remove"
  1093. font-size 10
  1094. post-spacing 6 }}
  1095. set items [list_extensions]
  1096. set possible_items [get_filtered_configs extensions]
  1097. set useful_items [list]
  1098. foreach item $items {
  1099. if {$item in $possible_items} {
  1100. lappend useful_items $item
  1101. }
  1102. }
  1103. set presentation [list]
  1104. foreach i $useful_items {
  1105. lappend presentation [utils::get_extension_display_name_by_config_name ${i}]
  1106. }
  1107. lappend menu_def presentation $presentation
  1108. return [prepare_menu_list $useful_items 10 $menu_def]
  1109. }
  1110. proc menu_remove_extension_exec {item} {
  1111. menu_close_all
  1112. remove_extension $item
  1113. }
  1114. proc menu_create_connectors_list {} {
  1115. set menu_def {
  1116. execute menu_connector_exec
  1117. font-size 8
  1118. border-size 2
  1119. width 200
  1120. xpos 100
  1121. ypos 120
  1122. header { text "Connectors"
  1123. font-size 10
  1124. post-spacing 6 }}
  1125. set items [machine_info connector]
  1126. set presentation [list]
  1127. foreach item $items {
  1128. set plugged [get_pluggable_for_connector $item]
  1129. set plugged_presentation ""
  1130. if {$plugged ne ""} {
  1131. set plugged_presentation " ([machine_info pluggable $plugged])"
  1132. }
  1133. lappend presentation "[machine_info connector $item]: $plugged$plugged_presentation"
  1134. }
  1135. lappend menu_def presentation $presentation
  1136. return [prepare_menu_list $items 5 $menu_def]
  1137. }
  1138. proc menu_connector_exec {item} {
  1139. menu_create [create_menu_pluggable_list $item]
  1140. select_menu_item [get_pluggable_for_connector $item]
  1141. }
  1142. proc create_menu_pluggable_list {connector} {
  1143. set menu_def [list \
  1144. execute [list menu_plug_exec $connector] \
  1145. font-size 8 \
  1146. border-size 2 \
  1147. width 200 \
  1148. xpos 110 \
  1149. ypos 140 \
  1150. header [list text "What to Plug into [machine_info connector $connector]?" \
  1151. font-size 10 \
  1152. post-spacing 6 ]]
  1153. set items [list]
  1154. set class [machine_info connectionclass $connector]
  1155. # find out which pluggables are already plugged
  1156. # (currently a pluggable can be used only once per machine)
  1157. set already_plugged [list]
  1158. foreach other_connector [machine_info connector] {
  1159. set other_plugged [get_pluggable_for_connector $other_connector]
  1160. if {$other_plugged ne "" && $other_connector ne $connector} {
  1161. lappend already_plugged $other_plugged
  1162. }
  1163. }
  1164. # get a list of all pluggables that fit this connector
  1165. # and which are not plugged yet in other connectors
  1166. foreach pluggable [machine_info pluggable] {
  1167. if {$pluggable ni $already_plugged && [machine_info connectionclass $pluggable] eq $class} {
  1168. lappend items $pluggable
  1169. }
  1170. }
  1171. set presentation [list]
  1172. foreach item $items {
  1173. lappend presentation "$item: [machine_info pluggable $item]"
  1174. }
  1175. set plugged [get_pluggable_for_connector $connector]
  1176. if {$plugged ne ""} {
  1177. set items [linsert $items 0 "--unplug--"]
  1178. set presentation [linsert $presentation 0 "Nothing, unplug $plugged ([machine_info pluggable $plugged])"]
  1179. }
  1180. lappend menu_def presentation $presentation
  1181. return [prepare_menu_list $items 5 $menu_def]
  1182. }
  1183. proc menu_plug_exec {connector pluggable} {
  1184. set command ""
  1185. if {$pluggable eq "--unplug--"} {
  1186. set command "unplug {$connector}"
  1187. } else {
  1188. set command "plug {$connector} {$pluggable}"
  1189. }
  1190. #note: NO braces around $command
  1191. if {[catch $command errorText]} {
  1192. osd::display_message $errorText error
  1193. } else {
  1194. menu_close_top
  1195. # refresh the connectors menu
  1196. # The list must be recreated, so menu_refresh_top won't work
  1197. menu_close_top
  1198. menu_create [menu_create_connectors_list]
  1199. }
  1200. }
  1201. proc menu_create_toys_list {} {
  1202. set menu_def {
  1203. execute menu_toys_exec
  1204. font-size 8
  1205. border-size 2
  1206. width 200
  1207. xpos 100
  1208. ypos 120
  1209. header { text "Toys and Utilities"
  1210. font-size 10
  1211. post-spacing 6 }}
  1212. set items [list]
  1213. set presentation [list]
  1214. # This also picks up 'lazy' command names
  1215. foreach cmd [openmsx::all_command_names] {
  1216. if {[string match toggle_* $cmd]} {
  1217. lappend items $cmd
  1218. lappend presentation [string map {_ " "} [string range $cmd 7 end]]
  1219. }
  1220. }
  1221. lappend menu_def presentation $presentation
  1222. return [prepare_menu_list $items 5 $menu_def]
  1223. }
  1224. proc menu_toys_exec {toy} {
  1225. return [$toy]
  1226. }
  1227. proc ls {directory extensions} {
  1228. set dirs [list]
  1229. set specialdir [list]
  1230. set items [list]
  1231. if {[catch {
  1232. set files [glob -nocomplain -tails -directory $directory -types {f r} *]
  1233. set items [lsearch -regexp -all -inline -nocase $files .*\\.($extensions)]
  1234. set dirs [glob -nocomplain -tails -directory $directory -types {d r x} *]
  1235. set specialdir [glob -nocomplain -tails -directory $directory -types {hidden d} ".openMSX"]
  1236. } errorText]} {
  1237. osd::display_message "Unable to read dir $directory: $errorText" error
  1238. }
  1239. set dirs2 [list]
  1240. foreach dir [concat $dirs $specialdir] {
  1241. lappend dirs2 "$dir/"
  1242. }
  1243. set extra_entries [list]
  1244. set volumes [file volumes]
  1245. if {$directory ni $volumes} {
  1246. # check whether .. is readable (it's not always so on Android)
  1247. if {[file readable [file join $directory ..]]} {
  1248. lappend extra_entries ".."
  1249. }
  1250. } else {
  1251. if {[llength $volumes] > 1} {
  1252. set extra_entries $volumes
  1253. }
  1254. }
  1255. return [concat [lsort $extra_entries] [lsort $dirs2] [lsort $items]]
  1256. }
  1257. proc is_empty_dir {directory extensions} {
  1258. set files [list]
  1259. catch {set files [glob -nocomplain -tails -directory $directory -types {f r} *]}
  1260. set items [lsearch -regexp -all -inline -nocase $files .*\\.($extensions)]
  1261. if {[llength $items] != 0} {return false}
  1262. set dirs [list]
  1263. catch {set dirs [glob -nocomplain -tails -directory $directory -types {d r x} *]}
  1264. if {[llength $dirs] != 0} {return false}
  1265. set specialdir [list]
  1266. catch {set specialdir [glob -nocomplain -tails -directory $directory -types {hidden d} ".openMSX"]}
  1267. if {[llength $specialdir] != 0} {return false}
  1268. return true
  1269. }
  1270. proc menu_create_rom_list {path slot} {
  1271. set menu_def [list execute [list menu_select_rom $slot] \
  1272. font-size 8 \
  1273. border-size 2 \
  1274. width 200 \
  1275. xpos 100 \
  1276. ypos 120 \
  1277. header { text "ROMs $::osd_rom_path" \
  1278. font-size 10 \
  1279. post-spacing 6 }]
  1280. set extensions "rom|ri|mx1|mx2|zip|gz"
  1281. set items [list]
  1282. set presentation [list]
  1283. if {[lindex [$slot] 2] ne "empty"} {
  1284. lappend items "--eject--"
  1285. lappend presentation "--eject-- [file tail [lindex [$slot] 1]]"
  1286. }
  1287. set i 1
  1288. foreach pool_path [filepool::get_paths_for_type rom] {
  1289. if {$path ne $pool_path && [file exists $pool_path] &&
  1290. ![is_empty_dir $pool_path $extensions]} {
  1291. lappend items $pool_path
  1292. lappend presentation "\[ROM Pool $i\]"
  1293. }
  1294. incr i
  1295. }
  1296. set files [ls $path $extensions]
  1297. set items [concat $items $files]
  1298. set presentation [concat $presentation $files]
  1299. lappend menu_def presentation $presentation
  1300. return [prepare_menu_list $items 10 $menu_def]
  1301. }
  1302. proc menu_select_rom {slot item {open_main false}} {
  1303. if {$item eq "--eject--"} {
  1304. menu_close_all
  1305. $slot eject
  1306. reset
  1307. } else {
  1308. set fullname [file join $::osd_rom_path $item]
  1309. if {[file isdirectory $fullname]} {
  1310. menu_close_top
  1311. set ::osd_rom_path [file normalize $fullname]
  1312. menu_create [menu_create_rom_list $::osd_rom_path $slot]
  1313. } else {
  1314. set mappertype ""
  1315. set hash [sha1sum $fullname]
  1316. if {[catch {set mappertype [dict get [openmsx_info software $hash] mapper_type_name]}]} {
  1317. # not in the database, execute after selecting mapper type
  1318. set menu_proc menu_create
  1319. if {$open_main} { set menu_proc do_menu_open }
  1320. ${menu_proc} [menu_create_mappertype_list $slot $fullname]
  1321. } else {
  1322. # in the database, so just execute
  1323. menu_rom_with_mappertype_exec $slot $fullname $mappertype
  1324. }
  1325. }
  1326. }
  1327. }
  1328. proc menu_rom_with_mappertype_exec {slot fullname mappertype} {
  1329. if {[catch {$slot $fullname -romtype $mappertype} errorText]} {
  1330. osd::display_message "Can't insert ROM: $errorText" error
  1331. } else {
  1332. menu_close_all
  1333. set rominfo [getlist_rom_info]
  1334. if {$rominfo eq ""} {
  1335. osd::display_message "No ROM information available..."
  1336. } else {
  1337. osd::display_message "Now running ROM:\nTitle:\nYear:\nCompany:\nCountry:\nStatus:\nRemark:"
  1338. append result " \n" \
  1339. "[dict get $rominfo title]\n" \
  1340. "[dict get $rominfo year]\n" \
  1341. "[dict get $rominfo company]\n" \
  1342. "[dict get $rominfo country]\n" \
  1343. "[dict get $rominfo status]\n"
  1344. if {[dict get $rominfo remark] ne ""} {
  1345. append result [dict get $rominfo remark]
  1346. } else {
  1347. append result "None"
  1348. }
  1349. set txt_size 6
  1350. set xpos 35
  1351. # TODO: prevent this from being duplicated from osd_widgets::text_box
  1352. if {$::scale_factor == 1} {
  1353. set txt_size 9
  1354. set xpos 53
  1355. }
  1356. # TODO: this code knows the internal name of the widget of osd::display_message proc... it shouldn't need to.
  1357. osd create text osd_display_message.rominfo_text -x $xpos -y 2 -size $txt_size -rgba 0xffffffff -text "$result"
  1358. }
  1359. reset
  1360. }
  1361. }
  1362. proc menu_create_mappertype_list {slot fullname} {
  1363. set menu_def [list execute [list menu_rom_with_mappertype_exec $slot $fullname] \
  1364. font-size 8 \
  1365. border-size 2 \
  1366. width 200 \
  1367. xpos 100 \
  1368. ypos 120 \
  1369. header { text "Select mapper type" \
  1370. font-size 10 \
  1371. post-spacing 6 }]
  1372. set items [openmsx_info romtype]
  1373. set presentation [list]
  1374. foreach i $items {
  1375. lappend presentation "[dict get [openmsx_info romtype $i] description]"
  1376. }
  1377. set items_sorted [list "auto"]
  1378. set presentation_sorted [list "Auto-detect (guess)"]
  1379. foreach i [lsort -dictionary -indices $presentation] {
  1380. lappend presentation_sorted [lindex $presentation $i]
  1381. lappend items_sorted [lindex $items $i]
  1382. }
  1383. lappend menu_def presentation $presentation_sorted
  1384. return [prepare_menu_list $items_sorted 10 $menu_def]
  1385. }
  1386. proc menu_create_disk_list {path drive} {
  1387. set menu_def [list execute [list menu_select_disk $drive] \
  1388. font-size 8 \
  1389. border-size 2 \
  1390. width 200 \
  1391. xpos 100 \
  1392. ypos 120 \
  1393. header { text "Disks $::osd_disk_path" \
  1394. font-size 10 \
  1395. post-spacing 6 }]
  1396. set cur_image [lindex [$drive] 1]
  1397. set extensions "dsk|zip|gz|xsa|dmk|di1|di2|fd?|1|2|3|4|5|6|7|8|9"
  1398. set items [list]
  1399. set presentation [list]
  1400. if {[lindex [$drive] 2] ne "empty readonly"} {
  1401. lappend items "--eject--"
  1402. lappend presentation "--eject-- [file tail $cur_image]"
  1403. }
  1404. set i 1
  1405. foreach pool_path [filepool::get_paths_for_type disk] {
  1406. if {$path ne $pool_path && [file exists $pool_path] &&
  1407. ![is_empty_dir $pool_path $extensions]} {
  1408. lappend items $pool_path
  1409. lappend presentation "\[Disk Pool $i\]"
  1410. }
  1411. incr i
  1412. }
  1413. if {$cur_image ne $path} {
  1414. lappend items "."
  1415. lappend presentation "--insert this dir as disk--"
  1416. }
  1417. set files [ls $path $extensions]
  1418. set items [concat $items $files]
  1419. set presentation [concat $presentation $files]
  1420. lappend menu_def presentation $presentation
  1421. return [prepare_menu_list $items 10 $menu_def]
  1422. }
  1423. proc menu_select_disk {drive item {dummy false}} {
  1424. if {$item eq "--eject--"} {
  1425. set cur_image [lindex [$drive] 1]
  1426. menu_close_all
  1427. $drive eject
  1428. osd::display_message "Disk $cur_image ejected from drive [get_slot_str $drive]!"
  1429. } else {
  1430. # if the item is already a directory, it's an absolute path, use that as fullname
  1431. if {[file isdirectory $item] && $item ne "." && $item ne ".."} {
  1432. set fullname $item
  1433. set abspath true
  1434. } else {
  1435. set fullname [file normalize [file join $::osd_disk_path $item]]
  1436. set abspath false
  1437. }
  1438. if {[file isdirectory $fullname] && $item ne "." && !$abspath} {
  1439. menu_close_top
  1440. set ::osd_disk_path [file normalize $fullname]
  1441. menu_create [menu_create_disk_list $::osd_disk_path $drive]
  1442. } else {
  1443. if {[catch {$drive $fullname} errorText]} {
  1444. osd::display_message "Can't insert disk: $errorText" error
  1445. } else {
  1446. menu_close_all
  1447. if {$item eq "."} { set item $fullname }
  1448. osd::display_message "Disk $item inserted in drive [get_slot_str $drive]!"
  1449. }
  1450. }
  1451. }
  1452. }
  1453. proc menu_create_tape_list {path} {
  1454. variable taperecordings_directory
  1455. set menu_def { execute menu_select_tape
  1456. font-size 8
  1457. border-size 2
  1458. width 200
  1459. xpos 100
  1460. ypos 120
  1461. header { text "Tapes $::osd_tape_path"
  1462. font-size 10
  1463. post-spacing 6 }}
  1464. set extensions "cas|wav|zip|gz"
  1465. set items [list]
  1466. set presentation [list]
  1467. lappend items "--create--"
  1468. lappend presentation "--create new and insert--"
  1469. set inserted [lindex [cassetteplayer] 1]
  1470. if {$inserted ne ""} {
  1471. lappend items "--eject--"
  1472. lappend presentation "--eject-- [file tail $inserted]"
  1473. lappend items "--rewind--"
  1474. lappend presentation "--rewind-- [file tail $inserted]"
  1475. }
  1476. if {$path ne $taperecordings_directory && [file exists $taperecordings_directory]} {
  1477. lappend items $taperecordings_directory
  1478. lappend presentation "\[My Tape Recordings\]"
  1479. }
  1480. set i 1
  1481. foreach pool_path [filepool::get_paths_for_type tape] {
  1482. if {$path ne $pool_path && [file exists $pool_path] &&
  1483. ![is_empty_dir $pool_path $extensions]} {
  1484. lappend items $pool_path
  1485. lappend presentation "\[Tape Pool $i\]"
  1486. }
  1487. incr i
  1488. }
  1489. set files [ls $path $extensions]
  1490. set items [concat $items $files]
  1491. set presentation [concat $presentation $files]
  1492. lappend menu_def presentation $presentation
  1493. return [prepare_menu_list $items 10 $menu_def]
  1494. }
  1495. proc menu_select_tape {item} {
  1496. variable taperecordings_directory
  1497. if {$item eq "--create--"} {
  1498. menu_close_all
  1499. osd::display_message [cassetteplayer new [menu_free_tape_name]]
  1500. } elseif {$item eq "--eject--"} {
  1501. menu_close_all
  1502. osd::display_message [cassetteplayer eject]
  1503. } elseif {$item eq "--rewind--"} {
  1504. menu_close_all
  1505. osd::display_message [cassetteplayer rewind]
  1506. } else {
  1507. set fullname [file join $::osd_tape_path $item]
  1508. if {[file isdirectory $fullname]} {
  1509. menu_close_top
  1510. set ::osd_tape_path [file normalize $fullname]
  1511. menu_create [menu_create_tape_list $::osd_tape_path]
  1512. } else {
  1513. if {[catch {cassetteplayer $fullname} errorText]} {
  1514. osd::display_message "Can't set tape: $errorText" error
  1515. } else {
  1516. osd::display_message "Inserted tape $item!"
  1517. menu_close_all
  1518. }
  1519. }
  1520. }
  1521. }
  1522. proc menu_free_tape_name {} {
  1523. variable taperecordings_directory
  1524. set existing [list]
  1525. foreach f [lsort [glob -tails -directory $taperecordings_directory -type f -nocomplain *.wav]] {
  1526. lappend existing [file rootname $f]
  1527. }
  1528. set i 1
  1529. while 1 {
  1530. set name [format "[guess_title untitled] %04d" $i]
  1531. if {$name ni $existing} {
  1532. return $name
  1533. }
  1534. incr i
  1535. }
  1536. }
  1537. proc menu_create_hdd_list {path drive} {
  1538. return [prepare_menu_list [ls $path "dsk|zip|gz|hdd"] \
  1539. 10 \
  1540. [list execute [list menu_select_hdd $drive]\
  1541. font-size 8 \
  1542. border-size 2 \
  1543. width 200 \
  1544. xpos 100 \
  1545. ypos 120 \
  1546. header { text "Hard disk images $::osd_hdd_path"
  1547. font-size 10
  1548. post-spacing 6 }]]
  1549. }
  1550. proc menu_select_hdd {drive item} {
  1551. set fullname [file join $::osd_hdd_path $item]
  1552. if {[file isdirectory $fullname]} {
  1553. menu_close_top
  1554. set ::osd_hdd_path [file normalize $fullname]
  1555. menu_create [menu_create_hdd_list $::osd_hdd_path $drive]
  1556. } else {
  1557. confirm_action "Really power off to change HDD image?" osd_menu::confirm_change_hdd [list $item $drive]
  1558. }
  1559. }
  1560. proc confirm_change_hdd {item result} {
  1561. menu_close_top
  1562. if {$result eq "Yes"} {
  1563. set fullname [file join $::osd_hdd_path [lindex $item 0]]
  1564. if {[catch {set ::power off; [lindex $item 1] $fullname} errorText]} {
  1565. osd::display_message "Can't change hard disk image: $errorText" error
  1566. # TODO: we already powered off even though the file may be invalid... save state first?
  1567. } else {
  1568. osd::display_message "Changed hard disk image to [lindex $item 0]!"
  1569. menu_close_all
  1570. }
  1571. set ::power on
  1572. }
  1573. }
  1574. proc menu_create_ld_list {path} {
  1575. set menu_def [list execute menu_select_ld \
  1576. font-size 8 \
  1577. border-size 2 \
  1578. width 200 \
  1579. xpos 100 \
  1580. ypos 120 \
  1581. header { text "LaserDiscs $::osd_ld_path" \
  1582. font-size 10 \
  1583. post-spacing 6 }]
  1584. set cur_image [lindex [laserdiscplayer] 1]
  1585. set extensions "ogv"
  1586. set items [list]
  1587. set presentation [list]
  1588. if {$cur_image ne ""} {
  1589. lappend items "--eject--"
  1590. lappend presentation "--eject-- [file tail $cur_image]"
  1591. }
  1592. set files [ls $path $extensions]
  1593. set items [concat $items $files]
  1594. set presentation [concat $presentation $files]
  1595. lappend menu_def presentation $presentation
  1596. return [prepare_menu_list $items 10 $menu_def]
  1597. }
  1598. proc menu_select_ld {item} {
  1599. if {$item eq "--eject--"} {
  1600. menu_close_all
  1601. osd::display_message [laserdiscplayer eject]
  1602. set cur_image [lindex [laserdiscplayer] 1]
  1603. osd::display_message "LaserDisc $cur_image ejected!"
  1604. } else {
  1605. set fullname [file join $::osd_ld_path $item]
  1606. if {[file isdirectory $fullname]} {
  1607. menu_close_top
  1608. set ::osd_ld_path [file normalize $fullname]
  1609. menu_create [menu_create_ld_list $::osd_ld_path]
  1610. } else {
  1611. if {[catch {laserdiscplayer insert $fullname} errorText]} {
  1612. osd::display_message "Can't load LaserDisc: $errorText" error
  1613. } else {
  1614. osd::display_message "Loaded LaserDisc $item!"
  1615. menu_close_all
  1616. }
  1617. }
  1618. }
  1619. }
  1620. proc get_savestates_list_presentation_sorted {} {
  1621. set presentation [list]
  1622. foreach i [lsort -integer -index 1 -decreasing [savestate::list_savestates_raw]] {
  1623. set pres_str [lindex $i 0]
  1624. lappend presentation $pres_str
  1625. }
  1626. return $presentation
  1627. }
  1628. proc menu_create_load_state {} {
  1629. set menu_def \
  1630. { execute menu_loadstate_exec
  1631. font-size 8
  1632. border-size 2
  1633. width 200
  1634. xpos 100
  1635. ypos 120
  1636. on-open {osd create rectangle "preview" -x 225 -y 5 -w 90 -h 70 -rgba 0x30303080 -scaled true}
  1637. on-close {osd destroy "preview"}
  1638. on-select menu_loadstate_select
  1639. on-deselect menu_loadstate_deselect
  1640. header { text "Load State"
  1641. font-size 10
  1642. post-spacing 6 }}
  1643. set items [list_savestates -t]
  1644. lappend menu_def presentation [get_savestates_list_presentation_sorted]
  1645. return [prepare_menu_list $items 10 $menu_def]
  1646. }
  1647. proc menu_create_save_state {} {
  1648. set items [concat [list "create new"] [list_savestates -t]]
  1649. set menu_def \
  1650. { execute menu_savestate_exec
  1651. font-size 8
  1652. border-size 2
  1653. width 200
  1654. xpos 100
  1655. ypos 120
  1656. on-open {osd create rectangle "preview" -x 225 -y 5 -w 90 -h 70 -rgba 0x30303080 -scaled true}
  1657. on-close {osd destroy "preview"}
  1658. on-select menu_loadstate_select
  1659. on-deselect menu_loadstate_deselect
  1660. header { text "Save State"
  1661. font-size 10
  1662. post-spacing 6 }}
  1663. lappend menu_def presentation [concat [list "create new"] [get_savestates_list_presentation_sorted]]
  1664. return [prepare_menu_list $items 10 $menu_def]
  1665. }
  1666. proc menu_loadstate_select {item} {
  1667. set png $::env(OPENMSX_USER_DATA)/../savestates/${item}.png
  1668. catch {osd create rectangle "preview.image" -relx 0.05 -rely 0.05 -w 80 -h 60 -image $png}
  1669. }
  1670. proc menu_loadstate_deselect {item} {
  1671. osd destroy "preview.image"
  1672. }
  1673. proc menu_loadstate_exec {item} {
  1674. if {[catch {loadstate $item} errorText]} {
  1675. osd::display_message $errorText error
  1676. } else {
  1677. menu_close_all
  1678. }
  1679. }
  1680. proc menu_savestate_exec {item} {
  1681. if {$item eq "create new"} {
  1682. set item [menu_free_savestate_name]
  1683. confirm_save_state $item "Yes"
  1684. menu_close_all
  1685. } else {
  1686. confirm_action "Overwrite $item?" osd_menu::confirm_save_state $item
  1687. }
  1688. }
  1689. proc confirm_save_state {item result} {
  1690. menu_close_top
  1691. if {$result eq "Yes"} {
  1692. if {[catch {savestate $item} errorText]} {
  1693. osd::display_message $errorText error
  1694. } else {
  1695. osd::display_message "State saved to $item!"
  1696. menu_close_all
  1697. }
  1698. }
  1699. }
  1700. proc menu_free_savestate_name {} {
  1701. set existing [list_savestates]
  1702. set i 1
  1703. while 1 {
  1704. set name [format "[guess_title savestate] %04d" $i]
  1705. if {$name ni $existing} {
  1706. return $name
  1707. }
  1708. incr i
  1709. }
  1710. }
  1711. proc confirm_action {text action item} {
  1712. set items [list "No" "Yes"]
  1713. set menu_def [list execute [list $action $item] \
  1714. font-size 8 \
  1715. border-size 2 \
  1716. width 210 \
  1717. xpos 100 \
  1718. ypos 100 \
  1719. header [list text $text \
  1720. font-size 10 \
  1721. post-spacing 6 ]]
  1722. osd_menu::menu_create [osd_menu::prepare_menu_list $items [llength $items] $menu_def]
  1723. }
  1724. proc menu_loadreplay_exec {item} {
  1725. if {[catch {reverse loadreplay $item} errorText]} {
  1726. osd::display_message $errorText error
  1727. } else {
  1728. menu_close_all
  1729. }
  1730. }
  1731. proc menu_loadscript_exec {item} {
  1732. if {[catch {source $item} errorText]} {
  1733. osd::display_message $errorText error
  1734. }
  1735. }
  1736. proc create_slot_actions_to_put_stuff_in_slot {slot path listtype} {
  1737. return [list actions [list A [list osd_menu::menu_select_$listtype $slot $path]]]
  1738. }
  1739. proc drop_handler { event } {
  1740. variable mediaslot_info
  1741. lassign $event type filename
  1742. set category [openmsx_info file_type_category $filename]
  1743. set isdir [file isdirectory $filename]
  1744. if {$category eq "unknown" && $isdir} {
  1745. set category "disk"
  1746. }
  1747. set filetext "file"
  1748. if {$isdir} {
  1749. set filetext "folder"
  1750. }
  1751. if {$category eq "disk" || $category eq "rom"} {
  1752. set longmediaslotname [dict get $mediaslot_info $category longmediaslotname]
  1753. set listtype [dict get $mediaslot_info $category listtype]
  1754. set mediabasecommand [dict get $mediaslot_info $category mediabasecommand]
  1755. set slots [lsort [info command ${mediabasecommand}?]]
  1756. if {[llength $slots] == 0} {
  1757. osd::display_message "Can't handle dropped $filetext $filename, no $longmediaslotname present." error
  1758. } elseif {[llength $slots] > 1} {
  1759. set path $filename
  1760. set menutitle "Select ${longmediaslotname}"
  1761. set create_action_proc "create_slot_actions_to_put_stuff_in_slot"
  1762. osd_menu::do_menu_open [create_slot_menu_def $slots $path $listtype $menutitle $create_action_proc]
  1763. } else {
  1764. osd_menu::menu_select_$listtype "${mediabasecommand}a" $filename true
  1765. }
  1766. } elseif {$category eq "laserdisc"} {
  1767. if {[info command laserdiscplayer] ne ""} {; # only exists on some Pioneers
  1768. osd_menu::menu_select_ld $filename
  1769. } else {
  1770. osd::display_message "Can't handle dropped $filetext $filename, no laser disc player present." error
  1771. }
  1772. } elseif {$category eq "cassette"} {
  1773. if {[catch "machine_info connector cassetteport"]} {; # example: turboR
  1774. osd::display_message "Can't handle dropped $filetext $filename, no cassette port present." error
  1775. } else {
  1776. osd_menu::menu_select_tape $filename
  1777. }
  1778. } elseif {$category eq "savestate"} {
  1779. osd_menu::menu_loadstate_exec [file rootname $filename]
  1780. } elseif {$category eq "replay"} {
  1781. osd_menu::menu_loadreplay_exec $filename
  1782. } elseif {$category eq "script"} {
  1783. osd_menu::menu_loadscript_exec $filename
  1784. } else {
  1785. # stuff we can implement outside openMSX
  1786. if {[file extension $filename] eq ".txt"} {
  1787. type_from_file $filename
  1788. } else {
  1789. osd::display_message "Don't know how to handle dropped $filetext $filename..." error
  1790. }
  1791. }
  1792. }
  1793. # Keep openmsx console from interfering with the osd menu:
  1794. # when the console is activated while the osd menu is already open, we want
  1795. # to prevent the osd menu from receiving the keys that are pressed in the
  1796. # console.
  1797. variable old_console $::console
  1798. proc console_input_layer {name1 name1 op} {
  1799. global console
  1800. variable old_console
  1801. if {$console == $old_console} return
  1802. set old_console $console
  1803. if {$console} {
  1804. activate_input_layer console -blocking
  1805. } else {
  1806. deactivate_input_layer console
  1807. }
  1808. }
  1809. trace add variable ::console write [namespace code console_input_layer]
  1810. namespace export main_menu_open
  1811. namespace export main_menu_close
  1812. namespace export main_menu_toggle
  1813. } ;# namespace osd_menu
  1814. namespace import osd_menu::*