#!/usr/bin/ruby # so my editor will like it... =begin Copyright 2010, Roger Pack This file is part of Sensible Cinema. Sensible Cinema is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Sensible Cinema is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Sensible Cinema. If not, see . =end print 'Loading Sensible Cinema...' require File.expand_path(File.dirname(__FILE__) + "/../lib/add_any_bundled_gems_to_load_path.rb") begin require 'sane' rescue LoadError p 'please run $ rake rebundle_copy_in_dependencies first, as we dislike needing locally installed rubygems' raise end raise 'need jruby' unless OS.java? raise 'need newer jruby!' unless RUBY_DESCRIPTION >= 'jruby 1.6.0' # may not need this since we still have to accomodate for so many oddities :P alias system_original system require_relative '../lib/mencoder_wrapper' require_relative '../lib/storage' require_relative '../lib/edl_parser' require_relative '../lib/mplayer_edl' require_relative '../lib/play_audio' require 'tmpdir' require_relative '../lib/swing_helpers' require_relative '../lib/drive_info' require 'whichr' require 'os' require 'ruby-wmi' unless OS.mac? vendor_cache = File.expand_path(File.dirname(__FILE__)) + '/../vendor/cache' if OS.windows? for name in ['.', 'mencoder', 'ffmpeg'] # put them all before the old path ENV['PATH'] = (vendor_cache + '/' + name).to_filename + ';' + ENV['PATH'] end installed_smplayer_folders = Dir['{c,d,e,f,g}:/program files*/MPlayer for Windows*'] # sometimes ends with UI? huh? for folder in installed_smplayer_folders ENV['PATH'] = ENV['PATH'] + ";#{folder.gsub('/', "\\")}" end else ENV['PATH'] = ENV['PATH'] + ':' + '/opt/local/bin' # add macports' bin in, just in case... end import 'javax.swing.ImageIcon' module SensibleSwing VERSION = File.read(File.dirname(__FILE__) + "/../VERSION").strip puts "v. " + VERSION UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()) # sigh class MainWindow < JFrame def new_jbutton title, only_on_create_mode, always_add = false button = JButton.new title button.set_bounds(44, @starting_button_y, @button_width, 23) if ARGV.index("--create-mode") always_add = true if only_on_create_mode else always_add = true if !only_on_create_mode end if always_add increment_button_location @panel.add button @buttons << button end button end def increment_button_location @starting_button_y += 30 end Storage = Storage.new("sensible_cinema") def initialize super "Sensible-Cinema #{VERSION} (GPL)" if !(Storage['main_license_accepted'] == VERSION) require_blocking_license_accept_dialog 'Sensible Cinema', 'gplv3', 'http://www.gnu.org/licenses/gpl.html', 'Sensible Cinema license agreement', "Sensible Cinema is distributed under the gplv3.\nBY CLICKING \"accept\" YOU SIGNIFY THAT YOU HAVE READ, UNDERSTOOD AND AGREED TO ABIDE BY THE TERMS OF THIS AGREEMENT" require_blocking_license_accept_dialog 'Sensible Cinema', 'is_it_legal_to_copy_dvds.txt file', File.expand_path(File.dirname(__FILE__) + "/../is_it_legal_to_copy_dvds.txt"), 'is_it_legal_to_copy_dvds.txt file', 'I acknowledge that I have read, understand, accept and agree to abide by the implications noted in the is_it_legal_to_copy_dvds.txt file' Storage['main_license_accepted'] = VERSION end setDefaultCloseOperation JFrame::EXIT_ON_CLOSE panel = JPanel.new @panel = panel @buttons = [] panel.set_layout nil add panel # why can't I just slap these down? jlabel = JLabel.new 'Welcome to Sensible Cinema!' happy = Font.new("Tahoma", Font::PLAIN, 11) jlabel.setFont(happy) jlabel.set_bounds(44,44,160,14) panel.add jlabel @starting_button_y = 120 @button_width = 400 ## BUTTONS ## @create_new_edl_for_current_dvd = new_jbutton("Create new Delete List for a DVD", true) @create_new_edl_for_current_dvd.on_clicked do create_brand_new_edl end @open_list = new_jbutton("Open/Edit a Delete List", true) @open_list.on_clicked { dialog = FileDialog.new(self, "Pick file to edit") dialog.set_directory EDL_DIR filename = dialog.go open_file_to_edit_it filename if filename } @create = new_jbutton( "Create edited copy of DVD on Your Hard Drive, from a DVD", false ) @create.on_clicked { do_copy_dvd_to_hard_drive false } @mplayer_edl = new_jbutton( "Watch DVD edited (realtime -- mplayer EDL)", false, true ) @mplayer_edl.on_clicked { do_mplayer_edl } @play_smplayer = new_jbutton( "Watch DVD unedited (realtime smplayer)", true).on_clicked { play_dvd_smplayer_unedited } @play_mplayer_raw = new_jbutton( "Watch DVD unedited (realtime mplayer--useful for subtitle timing)", true).on_clicked { play_dvd_smplayer_unedited true } # @watch_created_file = new_jbutton( "Watch edited file copy of DVD", false).on_clicked { # raise 'todo' # TODO # } @preview_section = new_jbutton( "Preview a certain time frame from file (edited)", true ) @preview_section.on_clicked { do_copy_dvd_to_hard_drive true } @preview_section_unedited = new_jbutton("Preview a certain time frame from file (unedited)", true) @preview_section_unedited.on_clicked { do_copy_dvd_to_hard_drive true, false, true } @rerun_preview = new_jbutton( "Re-run most recently watched file preview time frame", true ) @rerun_preview.on_clicked { repeat_last_copy_dvd_to_hard_drive } @fast_preview = new_jbutton( "fast preview all (smplayer EDL on file)", true).on_clicked { success, wrote_to_here_fulli = do_copy_dvd_to_hard_drive false, true sleep 0.5 # lodo take out ??? background_thread.join if background_thread # let it write out the original fulli, if necessary [?] nice_file = wrote_to_here_fulli #+ ".fast.mpg" if false#!File.exist?(nice_file) p = show_non_blocking_message_dialog("Creating quick lookup file--NB that for each changed deletion, you'll need to restart the fast preview SMplayer Also note that the start and end times will be slightly off if reality [delayed] Also note that while doing fast preview, it can be doing a normal preview as well in the background, simultaneously.") unless system_blocking("ffmpeg -i #{wrote_to_here_fulli} -target ntsc-dvd #{nice_file}") File.delete nice_file raise 'create ' end p.dispose # it will be active for sure end set_smplayer_opts "-edl #{EdlTempFile}" thread = do_mplayer_edl( "smplayer_portable #{nice_file}") # note the smplayer, but it's for the fast file... Thread.new { # XXX do we need this? begin thread.join ensure set_smplayer_opts '' end } } @watch_unedited = new_jbutton("Watch full DVD unedited (while simultaneously grabbing file to hard drive)", true) @watch_unedited.on_clicked { success_no_run, wrote_to_here_fulli = do_copy_dvd_to_hard_drive false, true, true sleep 5 unless success_no_run command = "smplayer_portable #{wrote_to_here_fulli}" system_non_blocking command } # until I can let them know *why* they would ever need this button... @display_unique = new_jbutton( "Display a DVD's unique ID", true ).on_clicked { drive, volume, dvd_id = choose_dvd_drive # display it, allow them to copy and paste it out get_user_input("#{drive} #{volume} for your copying+pasting pleasure (highlight, then ctrl+c to copy)\n This is USED to identify a disk to match it to its EDL, later.", "\"disk_unique_id\" => \"#{dvd_id}\",") } @upload = new_jbutton("Upload/E-mail suggestion/Submit anything", true).on_clicked { system_non_blocking("start mailto:sensible-cinema@googlegroups.com") system_non_blocking("start http://groups.google.com/group/sensible-cinema") } @progress_bar = JProgressBar.new(0, 100) @progress_bar.set_bounds(44,@starting_button_y,@button_width,23) @progress_bar.visible = false panel.add @progress_bar increment_button_location increment_button_location @exit = new_jbutton("Exit", false, true).on_clicked { #self.close # don't waste time :P kill_processes java.lang.System.exit 0 } increment_button_location increment_button_location setSize @button_width+80, @starting_button_y setIconImage(ImageIcon.new(__dir__ + "/../vendor/monkey.png").getImage()) check_for_dependencies end require 'fileutils' def set_smplayer_opts to_this smplayer_prefs_file = File.expand_path("~/.smplayer/smplayer.ini") old_prefs = File.read(smplayer_prefs_file) rescue '' old_prefs = "[advanced]\nmplayer_additional_options=" unless old_prefs.contain? 'mplayer_additional_options=' new_prefs = old_prefs.gsub(/mplayer_additional_options=.*/, "mplayer_additional_options=#{to_this}") FileUtils.mkdir_p File.dirname(smplayer_prefs_file) # case it doesn't exist' File.write(smplayer_prefs_file, new_prefs) p 'wrote', new_prefs.length, smplayer_prefs_file, to_this # no worky doze File.write(File.expand_path('~/.mplayer/config'), to_this) smplayer_prefs_file end def create_brand_new_edl drive, volume, dvd_id = choose_dvd_drive english_name = get_user_input("Enter a human readable DVD description for #{volume}", volume.gsub('_', ' ').downcase) input = <<-EOL # comments can go after a # on any line, for example this one. "name" => "#{english_name}", "mutes" => [ # example line, uncomment the leading "#" to make it active # "0:00:01.0", "0:00:02.0", "profanity", "da..", ], "blank_outs" => [ # example line, uncomment the leading "#" to make it active # "00:03:00.0" , "00:04:00.0", "violence", "of some sort", ], "volume_name" => "#{volume}", "disk_unique_id" => "#{dvd_id}", "dvd_title_track" => "1", # most DVD's use title 1. Not all do, though. If sensible-cinema plays anything except the main title (for example, a trailer), when you use it, see http://goo.gl/QHLIF to set this right. # "not edited out stuff" => "some violence", # "closing thoughts" => "still...", # "mplayer_dvd_splits" => ["59:59", "1:04:59"], # these are where, in mplayer, the DVD timestamp "resets" to zero for some reason (bug) http://goo.gl/yMfqX . [] is a valid option, if there are none. EOL # frame0 is the name of our window LOL tough to avoid that one... filename = EDL_DIR + "\\" + english_name.gsub(' ', '_') + '.txt' filename.downcase! File.write(filename, input) unless File.exist?(filename) # lodo let them choose name (?) open_file_to_edit_it filename end def system_blocking command, low_prio = false return true if command =~ /^@rem/ # jruby+MRI bug, I think... if low_prio out = IO.popen(command) # + " 2>&1" low_prio = 64 # from msdn if command =~ /(ffmpeg|mencoder)/ # XXXX not sure if there's a better way...because some *are* complex and have ampersands... # unfortunately have to check for nil because it could exit too early [?] exe_name = $1 + '.exe' begin p = proc{ ole = WMI::Win32_Process.find(:first, :conditions => {'Name' => exe_name}); sleep 1 unless ole; ole } piddy = p.call || p.call || p.call # we actually do need this to loop...guess we're too quick # but the first time through this still inexplicably fails... LODO if piddy # piddy.SetPriority low_prio # this call can seg fault at times...yikes...JRUBY-5422 pid = piddy.ProcessId # this doesn't seg fault, tho system_original("vendor\\setpriority -lowest #{pid}") # be able to use the PID on the command line else # XXXX first one always fails [?] huh? p 'unable to find to set priority ' + exe_name end rescue Exception => e p 'warning, got exception trying to set priority [jruby? ...]', e end end print out.read # let it finish out.close $?.exitstatus == 0 # 0 means success else raise command + ' failed' unless system_original command end end def system_non_blocking command Thread.new { system_original command } end # make them choose which system call to use explicitly undef system def download full_url, to_here require 'open-uri' writeOut = open(to_here, "wb") writeOut.write(open(full_url).read) writeOut.close end def _dbg require 'ruby-debug' debugger end def play_dvd_smplayer_unedited use_mplayer_instead = false # LODO really even if there are 2 it should still pick out the title track...hmm...on demand :P # lodo mac os x 'ify' drive, dvd_volume_name, dvd_id, edl_path_maybe_nil, descriptors_maybe_nil = choose_dvd_and_edl_for_it false if descriptors_maybe_nil title_track_maybe_nil = get_title_track(descriptors_maybe_nil, false) end if use_mplayer_instead exe = "mplayer " # maybe, maybe following would be useful for people just watching through the film (?) #"-font #{ENV['SystemRoot']}\\fonts\\ARIAL.TTF " else config_path = set_smplayer_opts " " # smplayer doesn't take -osdlevel 2 sniff... exe = "smplayer_portable -config-path \"#{File.dirname config_path}\" " end command = "#{exe} dvdnav://#{title_track_maybe_nil}/#{drive} -osd-fractions 2 -nocache " p command @play_smplayer_warn ||= show_non_blocking_message_dialog "Directions: 'o' key: turn on on-screen time stamps (note: these are 30 fps timestamps so will need to be converted to use). 'v' key: turn off subtitles. '.' key: step one frame. 'f' for full screen." system_non_blocking command end EdlTempFile = Dir.tmpdir + '/mplayer.temp.edl' def do_mplayer_edl play_this_mplayer = nil, add_secs_end = 0, add_secs_beginning = 0.5 drive, dvd_volume_name, dvd_id, edl_path, descriptors = choose_dvd_and_edl_for_it descriptors = EdlParser.parse_file edl_path splits = descriptors['mplayer_dvd_splits'] if splits == nil && !play_this_mplayer # don't display warning if they are watching the .fast file, since it doesn't need these if !play_this_mplayer show_blocking_message_dialog("warning: delete list does not contain mplayer replay information [mplayer_dvd_splits] so edits past a certain time period might not won't work ( http://goo.gl/yMfqX ).") end splits = [] end splits.map!{|s| EdlParser.translate_string_to_seconds(s) } edl_contents = MplayerEdl.convert_to_edl descriptors, add_secs_end, add_secs_beginning, splits # add a sec to mutes to accomodate for mplayer's oddness... File.write(EdlTempFile, edl_contents) title_track = get_title_track(descriptors) # oh the insanity of the console UI...LODO more user friendly player @popup ||= show_non_blocking_message_dialog("About to run mplayer DVD. To control it, use\n space for pause,\n f for full screen,\n arrow keys to seek.") # LODO dry up mplayer dvd opts... if OS.windows? play_this_mplayer ||= "mplayer -nocache dvdnav://#{title_track} " else # macports' mplayer has no dvdnav ?? play_this_mplayer ||= "mplayer dvd://#{title_track} " end if OS.windows? # direct3d for windows 7 old nvidia's sake yipes play_this_mplayer += " -vo direct3d " play_this_mplayer += " -font #{ENV['SystemRoot']}\\fonts\\ARIAL.TTF" end command = "#{play_this_mplayer} -framedrop -alang en -sid 1000 -edl #{File.expand_path EdlTempFile} -dvd-device #{drive}" # -hardframedrop might help but hurts just too much p command Thread.new { system_blocking command; @popup.dispose } end def require_blocking_license_accept_dialog program, license_name, license_url_should_also_be_embedded_by_you_in_message, title = 'Confirm Acceptance of License Agreement', message = nil puts 'Please confirm license agreement in open window.' old = ['no', 'yes', 'ok'].map{|name| 'OptionPane.' + name + 'ButtonText'}.map{|name| [name, UIManager.get(name)]} UIManager.put("OptionPane.yesButtonText", 'Accept') UIManager.put("OptionPane.noButtonText", "View License (#{license_name})") # cancel button stays the same... message ||= "Sensible Cinema requires a separately installed program (#{program}). You can install this program separately, or Sensible Cinema can download it for you. By clicking accept, below, you are confirming that you have read and agree to be bound by the terms of its license (the #{license_name}), located at #{license_url_should_also_be_embedded_by_you_in_message}. Click 'View License' to view it. If you do not agree to these terms, click 'Cancel'. You also agree that this is a separate program, with its own distribution, license, ownership and copyright. You agree that you are responsible for the download and use of this program, within sensible cinema or otherwise." returned = JOptionPane.showConfirmDialog nil, message, title, JOptionPane::YES_NO_CANCEL_OPTION # 1 is view button was clicked # 0 is accept # 2 is cancel if returned == 1 system_non_blocking("start #{license_url_should_also_be_embedded_by_you_in_message}") System.exit 1 end if returned == 2 p 'license not accepted...exiting' System.exit 1 end if returned == -1 p 'license exited early...exiting' System.exit 1 end p 'confirmed license noted: ' + license_name throw unless returned == 0 old.each{|name, old_setting| UIManager.put(name, old_setting)} end def print *args Kernel.print *args # avoid bin\sensible-cinema.rb:83:in `system_blocking': cannot convert instance of class org.jruby.RubyString to class java.awt.Graphics (TypeError) end def download_7zip Dir.mkdir('./vendor/cache') unless File.directory? 'vendor/cache' # development may not have it created yet... [?] unless File.exist? 'vendor/cache/7za.exe' Dir.chdir('vendor/cache') do print 'downloading unzipper (400K) ...' download("http://downloads.sourceforge.net/project/sevenzip/7-Zip/9.20/7za920.zip", "7za920.zip") system_blocking("../unzip.exe -o 7za920.zip") # -o means "overwrite" without prompting end end end def download_zip_to english_name, url, to_this download_7zip Dir.chdir('vendor/cache') do file_name = url.split('/')[-1] print "downloading #{english_name} ..." download(url, file_name) system_blocking("7za e #{file_name} -y -o#{to_this}") puts 'done ' + english_name # creates vendor/cache/mencoder/mencoder.exe... end end require 'lib/check_installed_mac.rb' def check_for_exe windows_loc, unix_name # in windows, that exe *at that location* must exist... if OS.windows? File.exist?(windows_loc) else if !CheckInstalledMac.check_for_installed(unix_name) exit 1 else true end end end def check_for_dependencies ffmpeg_exe_loc = File.expand_path('vendor/cache/ffmpeg/ffmpeg.exe') if !check_for_exe(ffmpeg_exe_loc, 'ffmpeg') require_blocking_license_accept_dialog 'ffmpeg', 'gplv2', 'http://www.gnu.org/licenses/gpl-2.0.html', "Appears that you need to install a dependency: ffmpeg." download_zip_to "ffmpeg (5MB)", "http://ffmpeg.zeranoe.com/builds/win32/shared/ffmpeg-git-1929807-win32-shared.7z", "ffmpeg" end if !check_for_exe('vendor/cache/mencoder/mencoder.exe', 'mencoder') require_blocking_license_accept_dialog 'mplayer', 'gplv2', 'http://www.gnu.org/licenses/gpl-2.0.html', "Appears that you need to install a dependency: mplayer with mencoder." download_zip_to "Mplayer/mencoder (6MB)", "http://downloads.sourceforge.net/project/mplayer-win32/MPlayer%20and%20MEncoder/revision%2033574/MPlayer-rtm-svn-33574.7z", "mencoder" end if ARGV.index('--create-mode') # they're going to want these dependencies if OS.windows? path = RubyWhich.new.which('smplayer_portable') if(path.length == 0) # this one has its own installer... show_blocking_message_dialog("It appears that you need to install a dependency: MPlayer for Windows (MPUI).\n Click ok to be directed to its download website, where you can download and install it (recommend: MPUI Full-package), then restart sensible cinema.", "Lacking dependency", JOptionPane::ERROR_MESSAGE) system_non_blocking("start http://mulder.dummwiedeutsch.de/#mplayer") # LODO would launchy help/work here with the full url? System.exit(1) end else check_for_exe("mplayer", "mplayer") # OS X ... puts 'warning for OS X users--the smplayer-type buttons aren\'t going to work' end end end def open_file_to_edit_it filename system_non_blocking "notepad \"#{filename}\"" end def single_edit_list_matches_dvd dvd_id matching = Dir[EDL_DIR + '/*.txt'].select{|file| begin parse_edl(file)["disk_unique_id"] == dvd_id rescue SyntaxError # ignore poorly formed delete lists for the auto choose p 'warning, unable to parse:' + file false end } if matching.length == 1 file = matching[0] p "selecting the one matching file #{file} #{dvd_id}" file else nil end end EDL_DIR = File.expand_path(__dir__ + "/../zamples/edit_decision_lists/dvds").to_filename def repeat_last_copy_dvd_to_hard_drive generate_and_run_bat_file *Storage['last_params'] end def new_filechooser JFileChooser.new end def show_blocking_message_dialog(message, title = message.split("\n")[0], style= JOptionPane::INFORMATION_MESSAGE) JOptionPane.showMessageDialog(nil, message, title, style) true end # call dispose on this to close it if it hasn't been canceled yet... def show_non_blocking_message_dialog message, close_button_text = 'Close' # lodo NonBlockingDialog it can get to the top instead of being so buried... NonBlockingDialog.new(message, close_button_text) end include_class javax.swing.UIManager def get_user_input(message, default = '') received = JOptionPane.showInputDialog(message, default) raise 'user canceled' unless received received end def parse_edl path EdlParser.parse_file path end def get_freespace path JFile.new(File.dirname(path)).get_usable_space end def choose_dvd_and_edl_for_it choose_file = true drive, dvd_volume_name, dvd_id = choose_dvd_drive unless @_edit_list_path edit_list_path = single_edit_list_matches_dvd(dvd_id) if !edit_list_path && choose_file fc = FileDialog.new(self) fc.set_title "Please pick a DVD Delete List File (non matching found)" fc.set_directory EDL_DIR edit_list_path = fc.go raise 'cancelled' unless edit_list_path end @_edit_list_path = edit_list_path end if @_edit_list_path # reload it just in case it has changed on disk descriptors = nil while(!descriptors) begin descriptors = parse_edl @_edit_list_path rescue SyntaxError => e puts e show_blocking_message_dialog("this file has an error--please fix then hit ok: \n" + @_edit_list_path + "\n " + e) end end end [drive, dvd_volume_name, dvd_id, @_edit_list_path, descriptors] end def get_title_track descriptors, use_default_of_one = true given = descriptors["dvd_title_track"] given ||= "1" if use_default_of_one given end def get_save_to_filename dvd_title @_get_save_to_filename ||= begin fc = new_filechooser fc.set_title "Pick where to save #{dvd_title} edited to" save_to_file_name = dvd_title + ' edited version' save_to_file_name = save_to_file_name.gsub(' ', '_').gsub( /\W/, '') # no punctuation or spaces for now... fc.set_file(get_drive_with_most_space_with_slash + save_to_file_name) save_to = fc.go a = File.open(File.dirname(save_to) + "/test_file_to_see_if_we_have_permission_to_write_to_this_folder", "w") a.close File.delete a.path freespace = get_freespace(save_to) if freespace < 16_000_000_000 show_blocking_message_dialog("Warning: there may not be enough space on the disk for #{save_to} (depending on DVD size, you may need like 16G free, but typically will need around 10GB free--you have #{freespace/1_000_000_000}GB free). Click OK to continue.") end raise 'cannot save to filname with spaces yet (ask for it)' if save_to =~ / / save_to end end def do_copy_dvd_to_hard_drive should_prompt_for_start_and_end_times, exit_early_if_fulli_exists = false, watch_unedited = false drive, dvd_volume_name, dvd_id, edit_list_path, descriptors = choose_dvd_and_edl_for_it descriptors = parse_edl(edit_list_path) if watch_unedited descriptors['mutes'] = descriptors['blank_outs'] = [] end # LODO allow for spaces in the save_to filename if should_prompt_for_start_and_end_times # only show this message once :) @show_block ||= show_blocking_message_dialog("Ok, let's preview just a portion of it. \nNote that you'll want to preview a section that wholly includes a deleted section in it\n For example, if it mutes from second 1 to second 10, you'll want to play from 00:00 to 00:12 or what not.\nAlso note that the first time you preview a section of a video, it will take a long time (like an hour) as it sets up the video for previewing.\nSubsequent previews will be faster, though, as long as you use the same filename.\n Also note that if you change your delete list, you'll need to close, and regenerate the video to see it with your new settings.", "Preview") old_start = Storage['start_time'] start_time = get_user_input("At what point in the video would you like to start your preview? (like 01:00 for starting at 1 minute)", Storage['start_time']) default_end = Storage['end_time'] if start_time and start_time != old_start default_end = EdlParser.translate_string_to_seconds(start_time) + 10 default_end = EdlParser.translate_time_to_human_readable(default_end) end end_time = get_user_input("At what point in the video would you like to finish your preview? (like 02:00 for ending at the 2 minute mark)", default_end) unless start_time and end_time JOptionPane.showMessageDialog(nil, " Please choose start and end", "Failed", JOptionPane::ERROR_MESSAGE) return end Storage['start_time'] = start_time Storage['end_time'] = end_time end dvd_title = descriptors['name'] || dvd_volume_name save_to = get_save_to_filename dvd_title fulli = MencoderWrapper.calculate_final_filename save_to if exit_early_if_fulli_exists if File.exist? fulli + ".done" return [true, fulli] end # make it create a dummy response file for us :) start_time = "00:00" end_time = "00:01" end dvd_title_track = get_title_track(descriptors) should_run_mplayer = should_prompt_for_start_and_end_times || exit_early_if_fulli_exists require_deletion_entry = true unless watch_unedited generate_and_run_bat_file save_to, edit_list_path, descriptors, drive, dvd_title, start_time, end_time, dvd_title_track, should_run_mplayer, require_deletion_entry [false, fulli] # false means it's running in a background thread :P end def get_drive_with_most_space_with_slash DriveInfo.get_drive_with_most_space_with_slash end # stubbable :) def get_mencoder_commands descriptors, drive, save_to, start_time, end_time, dvd_title_track, require_deletion_entry delete_partials = true unless start_time MencoderWrapper.get_bat_commands descriptors, drive, save_to, start_time, end_time, dvd_title_track, delete_partials, require_deletion_entry # delete partials... end def generate_and_run_bat_file save_to, edit_list_path, descriptors, drive, dvd_title, start_time, end_time, dvd_title_track, run_mplayer, require_deletion_entry Storage['last_params'] = [save_to, edit_list_path, descriptors, drive, dvd_title, start_time, end_time, dvd_title_track, run_mplayer, require_deletion_entry] begin commands = get_mencoder_commands descriptors, drive, save_to, start_time, end_time, dvd_title_track, require_deletion_entry rescue MencoderWrapper::TimingError => e show_blocking_message_dialog("Appears you chose a time frame with no deletion segment in it--please try again:" + e) return rescue Errno::EACCES => e show_blocking_message_dialog("Appears a file on the system is locked: perhaps you need to close down some instance of mplayer?" + e) return end temp_dir = Dir.tmpdir temp_file = temp_dir + '/vlc.temp.bat' File.write(temp_file, commands) popup = show_non_blocking_message_dialog( "Applying #{File.basename edit_list_path} \n against #{drive} (#{dvd_title}).\n" + "Copying to #{save_to}.\n" + "This could take quite awhile, and will prompt you and chime a noise when it is done.\n" + "You can close this window and continue working while it runs in the background.\n" + "NB that the created file will be playable only with VLC (possibly with smplayer, possibly with\n" + "Windows Media Player if you install ffdshow first with mpeg2 video checkbox checked.).", "OK") # allow our popups to still be serviced while it is running @background_thread = Thread.new { run_create_commands commands, save_to, run_mplayer popup.dispose } # LODO warn if they will overwrite a file in the end... end attr_accessor :background_thread, :buttons def run_create_commands batch_commands, save_to, run_mplayer @buttons.each{|b| b.set_enabled false} success = true lines = batch_commands.lines.to_a total_size = lines.length.to_f @progress_bar.visible=true @progress_bar.set_value(10) # start at 10% always, so they can see something. lines.each_with_index{|line, idx| if success puts "running #{line}" success = system_blocking(line, true) if !success puts "\n", 'line failed: ' + line + "\n" + ' see troubleshooting section in README.txt file! ignoring further processing commands...' end end @progress_bar.set_value(10 + idx/total_size*90) } @progress_bar.visible=false @buttons.each{|b| b.set_enabled true} if success saved_to = save_to + '.avi' if run_mplayer set_smplayer_opts '' system_non_blocking "smplayer_portable #{saved_to}" else show_file = "explorer /e,/select,\"#{File.expand_path(saved_to).to_filename}\"" system_blocking show_file # returns immediately PlayAudio.play(File.expand_path(File.dirname(__FILE__)) + "/../vendor/music.wav") show_blocking_message_dialog "Done--you may now watch file\n #{saved_to}\n in VLC player (or possibly smplayer)" end else show_blocking_message_dialog("Failed--please examine console output and report back!\nAlso consult the troubleshooting section of the README file.", "Failed", JOptionPane::ERROR_MESSAGE) end end # returns e:\, volume, dvd_id def choose_dvd_drive opticals = DriveInfo.get_dvd_drives_as_openstruct if @saved_opticals == opticals # memoize...kind of :) return @_choose_dvd_drive end unless opticals.find{|d| d.VolumeName } show_blocking_message_dialog 'insert a dvd first' exit 1 end names = opticals.map{|d| d.Name + "\\" + " (" + (d.VolumeName || 'Insert DVD to use') + ")"} if opticals.length != 1 count = 0 opticals.each{|d| count += 1 if d.VolumeName} if count == 1 # just choose it if there's only one disk in there p 'selecting only disk present in the various DVD drives' selected_idx = opticals.index{|d| d.VolumeName} else dialog = GetDisk.new(self, names) dialog.setSize 200,125 dialog.show selected_idx = dialog.selected_idx end else selected_idx = 0 p 'selecting user\'s only DVD drive ' + names[0] end if selected_idx disk = opticals[selected_idx] dvd_id = DriveInfo.md5sum_disk(disk.MountPoint) @_choose_dvd_drive = [disk.MountPoint, opticals[selected_idx].VolumeName, dvd_id] @saved_opticals = opticals # since we force them to have their disc already in there...guess this is ok for os x :) return @_choose_dvd_drive else puts 'did not select a drive...hard exiting' java.lang.System.exit 1 end end end class GetDisk < JDialog attr_reader :selected_idx def initialize parent, options_array super parent, true box = JComboBox.new box.add_action_listener do |e| idx = box.get_selected_index if idx != 0 # don't count choosing the first as a real entry @selected_idx = box.get_selected_index - 1 dispose end end box.add_item "Click to select DVD drive" # put something in index 0 options_array.each{|drive| box.add_item drive } add box pack end end end class ShutdownHook include java.lang.Runnable def initialize( &block) super() @block=block end def run @block[] end end def at_exit2( &block) hook = ShutdownHook.new( &block) java.lang.Runtime.getRuntime.addShutdownHook(java.lang.Thread.new( hook )) end def kill_processes if OS.windows? system_original("taskkill /f /im mencoder.exe 2>NUL") # todo...is there a better way? system_original("taskkill /f /im ffmpeg.exe 2>NUL") system_original("taskkill /f /im smplayer_portable.exe 2>NUL") system_original("taskkill /f /im mplayer.exe 2>NUL") end end at_exit2 { kill_processes # just in case } if $0 == __FILE__ if ARGV.index('-h') || ARGV.index('--help') puts 'syntax: [--create-mode]' else a = SensibleSwing::MainWindow.new a.set_visible true puts 'Please use the Sensible Cinema GUI window popup...' end end # icon derived from: http://www.threes.com/index.php?option=com_content&view=article&id=1800:three-wise-monkeys&catid=82:mythology&Itemid=62