Auto applying a template between loader and saver

Moderator: SecondMan

User avatar
ShadowMaker SdR
Fusionista
Posts: 655
Joined: Sun Sep 21, 2014 6:17 am
Answers: 4
Been thanked: 17 times

Re: Auto applying a template between loader and saver

#16

Post by ShadowMaker SdR » Wed Aug 22, 2018 11:23 am

Bryan, you're much more versed in scripting than I am. Can you look over the script I put in my post and see what obvious changes there might have been since its release that prevents it from working in 9.x? It was a really, really useful script and I think it might be quicker to revive/adapt it, than starting from scratch. But unfortunately I have no idea where to start myself.

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#17

Post by Midgardsormr » Wed Aug 22, 2018 12:18 pm

I don't know if this works. At least it doesn't throw errors any more. We don't use Fusion's render manager, and I'm unwilling to reconfigure my workstation when I don't know what I'm doing, so it didn't actually convert anything for me.

Code: [Select all] [Expand/Collapse] [Download] (Convert Footage.lua)
  1. ------------------------------------------------------------------------------------------------
  2. --      Convert Footage.eyeonscript,Revision 8
  3. --
  4. --
  5. --      Comp script for Eyeon Fusion or Rotation
  6. --
  7. --      by  peter@wwfx.net
  8. --
  9. --      Created Jun 2004
  10. --      Update Nov 03 2005 - Port to Fusion 5
  11. --      Update Jul 20 2006 - include some utility functions and added convertion between all known to fusion fileformats
  12. --      Update Aug 02 2006 - can work without templates, switching views in verbose mode.
  13. --      Update Sep 27 2006 - fixed fusion crash after creating the flows.
  14. --      Update Aug 09 2007 - Can use Fusion Slave or another Full Fusion on the network, to assemble and render comps.
  15. --                 - Can be launched from Rotation and Learning Edition.
  16. --              Update Aug 06 2010 - fixed fusion crash after creating the flows. The last created flow will remain active. Results are printed in console of comp from which script was executed.
  17. --      Update Aug 22 2017 - Port to Fusion 8 by Bryan Ray
  18. --
  19. --      requires:   eyeon.scriptlib, Rotation or Fusion LE
  20. --
  21. ------------------------------------------------------------------------------------------------
  22.  
  23. --[[
  24.  
  25. The script was made to automate creating of reference clips from firewire drives with film footage.
  26. It can be used to search recursive(or not) any folder for footage that Fusion can load
  27. and convert it to any other file format fusion can write to.
  28.  
  29. Some features:
  30.  
  31.     - Template comps. You can to create custom template comps, adjust bc,size,e.t.c. and use them for batch conversions.
  32.     - Can store specific loader's settings and apply them to loaders in generated comps. This is done via list of loader inputs which can be edited (arround line 530, table "keep").
  33.     - Choice for names of new clips and comps. They can be either source sequence name or its folder. Checks for duplicate names and append parent folder to name until it become unique.
  34.     - choosing queue group for created comps and submit them to render master.
  35.     - In verbose mode - current processed footage is displayed on fusion views, while creating the comp..
  36.     - Can create single jpg still from the middle of the sequence.(i.e. for web thumbs)
  37.  
  38. ------------------------------------------------------------------------------------------------
  39.  
  40.     INSTALL
  41.  
  42.     1.  Place Convert footage.eyeonscript in your fusions script directory.
  43.     2.  If using templates - path to template comps must be provided  in ('templates_path' variable)
  44.  
  45. ------------------------------------------------------------------------------------------------
  46.  
  47.     USAGE
  48.  
  49. You can use this without woring about templates, e.t.c. Templates are valuable, when loader must have some special predefined options
  50. i.e, ByPass convertion,spec import modes, also when resizing source footage or modifying it with bc..
  51. When not using templates, simple loader->saver comp will be generated, optionaly with framecounter.
  52. In verbose mode  script prints more info about what is hapening and also shows current sequence in fu views, but that takes more time to finish.
  53. When generating names for new comps and saver clips, there is a check to escape duplicate names.
  54. Name of next parent folder is included to clip name, until name becomes uniqie.
  55. i.e. scanning c:\my docs    for *.jpg and we have found those:
  56.     c:\my docs\folder.jpg
  57.     c:\my docs\band\folder.jpg
  58. I this case output comp and clip names will be set to:
  59.     folder.jpg
  60.     band_folder.jpg
  61. This is only if script find duplicate clip names.
  62. ------------------------------------------------------------------------------------------------
  63.  
  64.     CREATING NEW TEMPLATES
  65.  
  66. Provided template can be customized assuming existing tool names are not changed:
  67. LDR,SQ,JPG,MergeCounter,Counter - those are required by the script.
  68. If you dont need counter, script will work without it and its merge.
  69. The most important to keep are the loader(LDR) and savers(SQ,JPG). Between them you can put any other tools you need.
  70.  
  71. Place new templates in template folder - they will be listed in script dialog so you can choose them from there.
  72.  
  73. If you need to make two quicktimes, leave SQ to be set by script and add saver called MOV, point it to directory of your choice. Script will change only the clipname in the second saver.
  74.  
  75. ]]
  76.  
  77.  
  78. --=========================       CONFIGURATION START    =========================--
  79.  
  80. templates_path = [[X:\rnd\_defaults\templates\fus\]]
  81. --templates_path = Comp:MapPath("Comps:") .. "[ Templates ]" -- i.e. C:\Program Files\Fusion\Comps\[ Templates ]
  82.  
  83. --=========================     CONFIGURATION   END     =========================--
  84.  
  85.  
  86.  
  87. if (not comp) then
  88.     print("--\nConvert Footage is COMPOSITION SCRIPT.\n"
  89.     .."To use it, please move it to:\n"
  90.     ..Fusion():GetPrefs().Global.Paths.Scripts.."\\Comp"
  91.     .."\nopen new comp and look for \"Convert Footage\" in the script menu :)\n--")
  92.     do
  93.         return
  94.     end
  95. end
  96. local addslash  =   function (th)
  97.     if string.sub(th, - 1, - 1) ~= "\\" then
  98.         return
  99.         (th.. "\\")
  100.     else
  101.         return
  102.         th
  103.     end
  104. end
  105. local GetFormats    =   function()
  106.     formats,result,movs = FusionRemote:GetRegList(CT_ImageFormat),{},{}
  107.     for ik  in  pairs(formats)      do
  108.             f,ismovie = formats[ik]:GetAttrs().REGST_MediaFormat_Extension,
  109.                         formats[ik]:GetAttrs().REGB_MediaFormat_CanLoadMulti
  110.             if f then
  111.                 to = table.getn(f)
  112.                 for mm, to in pairs(f) do
  113.                     if (ismovie == true) then
  114.                         table.insert(movs, f[mm])
  115.                     end
  116.                     table.insert(result, f[mm])
  117.                 end
  118.             end
  119.         end
  120.         table.sort(result)
  121.     return
  122.     result, movs
  123. end
  124. local file_exists   =   function(th)  --    for unc, local dirs and files
  125.     if not th then
  126.         return false
  127.     end
  128.     if fileexists(th) then
  129.         return true
  130.     end
  131.     if string.sub(th, - 1, - 1) == "\\" then
  132.          b,s=(string.sub(th, 1, string.len(th) - 1)),(th)
  133.     else
  134.          b,s=th,th.."\\"
  135.     end
  136.     return(direxists(b) or direxists(s))
  137. end
  138. local pm = function(message)    print(string.rep('-', 50),("\n\t" .. message.."\n"))  end
  139. function ScanDir(dir, mask, recursive)
  140.     dir,    maxlen, clips = addslash(dir),  (maxlen or 0),  (clips or {})
  141.     local files = readdir(dir .. '*')
  142.     local tt,   temp = table.getn(files),   {}
  143.     if (tt > maxlen) then
  144.         maxlen = tt
  145.     end
  146.     for i = 1, tt do
  147.         --fi,       e = string.lower(dir .. files[i].Name),     nil
  148.         fi,     e = dir .. files[i].Name,   nil
  149.         if (files[i].IsDir == true and recursive == 1 and files[i].IsSystem == false) then
  150.             ScanDir(fi, mask, recursive)
  151.         else
  152.             string.gsub(fi, "^(.+)(%..+)$", function(n, E) e = E end)
  153.             if (fi and e) then
  154.                 if ((e == mask) or (mask == '*')) then
  155.                     local seq = eyeon.parseFilename(fi)
  156.                     if (eyeon.isin(movs, mask) == true) then
  157.                         t = seq.FullPath
  158.                     else
  159.                         t = seq.CleanName
  160.                     end
  161.                     if (eyeon.isin(temp, t) == false) then
  162.                         printf("%0s %8s", '->', seq.Path .. seq.FullName)
  163.                         table.insert(clips, seq)
  164.                         table.insert(temp, t)
  165.                         t = nil
  166.                     end
  167.                 end
  168.             end
  169.         end
  170.     end
  171.     scanned = (scanned or 0) + 1
  172. end
  173. myscriptprefs = function(op)
  174.  
  175.     local _scriptname=eyeon.split(debug.getinfo(1).source,[[/]])
  176.     local _scriptname=_scriptname[table.getn(_scriptname)]
  177.     local prefname  =   _scriptname..[[.ScriptPrefs]]
  178.  
  179.     cpath   =   fusion:MapPath("Profile:")..([[ScriptPrefs\]])
  180.  
  181.     local cfg       =   cpath..prefname
  182.    
  183.     pm("Using: "..cfg)
  184.    
  185.     if op and op=='write' then
  186.       if not fileexists(cpath) then
  187.         createdir(cpath)
  188.       end
  189.     file,err = io.open(cfg, "w + ")
  190.     if file then
  191.       local h="-- Script preferences for ".._scriptname.."\n-- generated by ".._scriptname.."\n\n"
  192.           file:write(h .. "_cfs = {}\n")
  193.           for i, v in ipairs(_cfs) do
  194.               if (type(v) == "string") then
  195.                   file:write("_cfs[\"" .. i .. "\"]= [["..v.."]]\n")
  196.               elseif (type(v) == "boolean" or type(v) == "number") then
  197.                   file:write("_cfs[\"" .. i .. "\"]= " .. v .. "\n")
  198.               end
  199.           end
  200.       file:close()
  201.       end
  202.       file = nil
  203.     else
  204.       if (fileexists(cfg)) then
  205.           dofile(cfg)
  206.           if (type("_cfs") == nil) then
  207.               return
  208.               {}
  209.           end
  210.       else
  211.           return
  212.           {}
  213.       end
  214.     end
  215. end
  216. function alert(message)
  217.     _ALERT(message)
  218.     ret = comp:AskUser("Alert!", {{"", "Text", Wrap=false, Lines=15,Default = message,Width=2}})
  219.     return ret
  220. end
  221. function getToolPositions(c)  --Fusion 5.2 and up. returns nil if 5.1 or lower
  222.     local positions =   {}
  223.     if not c.CurrentFrame then do return nil end end
  224.     local grid      =   c.CurrentFrame.FlowView
  225.     if not grid then do return nil end end
  226.     local grattr    =   grid:GetAttrs()
  227.     if grattr.VIEWN_Top then
  228.         grid:SetScale(0.5)
  229.         for i, tool in ipairs(c:GetToolList()) do
  230.             name    =   tostring(tool:GetAttrs().TOOLS_Name)
  231.             x,y =   c.CurrentFrame.FlowView:GetPos(tool)
  232.             positions[name]={["x"]=x,["y"]=y}
  233.         end
  234.     return positions
  235.     end
  236.  return nil
  237. end
  238. --========================   end utils
  239. local fta=fusion:GetAttrs()
  240. _cfs={}
  241. pm('local: '..fta.FUSIONS_FileName,' version ',fta.FUSIONS_Version,' with ',_G["_VERSION"])
  242. LocalIsFusion       =   eyeon.isin(split(fta.FUSIONS_FileName,'\\'),'Fusion.exe') and not (fta.FUSIONB_IsDemo==true)
  243. remote = comp:AskUser("", {
  244. {"", "Text", Wrap=true, Lines=3,Default ='Remote or local hostname/IP address with  Fusion or Render Slave to use:',Width=1},
  245. {"host", "Text", Wrap=false, Lines=1,Default ='localhost',Width=1},
  246. })
  247. if not remote then
  248. alert('Script aborted.')
  249.     do return end
  250. end
  251. FusionRemote=Fusion(remote['host'])
  252. if not FusionRemote then
  253.     alert('Can\'t connect to fusion instance')
  254.     do return end
  255. end
  256. local fa=FusionRemote:GetAttrs()
  257. pm(fa.FUSIONS_FileName,' version ',fa.FUSIONS_Version,' with ',_G["_VERSION"])
  258. local o=split(fa.FUSIONS_FileName,'\\')
  259. IsRotation  =   eyeon.isin(o,'Rotation.exe')
  260. IsSlave     =   eyeon.isin(o,'RenderSlave.exe') or eyeon.isin(o,'ConsoleSlave.exe')
  261. IsFusion    =   eyeon.isin(o,'Fusion.exe')
  262. IsVision    =   eyeon.isin(o,'Vision.exe')
  263. --========================
  264. ccomp = fusion.GetCurrentComp() or fusion:GetCurrentComp()
  265. frame = ccomp:GetFrameList()[1]
  266. frame:SwitchMainView('ConsoleView')  -- take users attention.
  267. _cfs,   templates_path, tpl = myscriptprefs('read'),    fusion:MapPath(addslash(eyeon.trim(templates_path))),   {}
  268. local exts,movs = GetFormats()
  269. if table.getn(exts)==0 then
  270. alert('Can\'t get list of file formats, aborting the srcipt!')
  271. do return end
  272. end
  273. exts2=exts
  274. exts2[table.getn(exts) + 1] = '- none -'
  275. if (file_exists(templates_path) == false) then
  276.     _cfs.ut = 0
  277.     _ale=[[The path to templates:\n
  278. --------------------------------------------------\n
  279. ]] .. (templates_path or '') .. [[
  280. --------------------------------------------------\n
  281. set in the beginig of this script is NOT VALID!
  282. If you want to use your own templates:
  283. [1].    Configure script :
  284. ]]..debug.getinfo(1).source..[[
  285. by editing it with text editor.
  286. Type there the path to templates.
  287. [2].    Run the script without templates once.
  288. Simple loader->merge->counter->saver comps will be generated.
  289. Copy those to your templates path. Adjust them to suit your needs,
  290. without changing names of existing tools, so script can find and set them later.
  291. [3]. Run the script again - you should see menu with your template comps.
  292. If you continue script will continue
  293. with creating basic template with loader->saver
  294. ]]
  295.     alert(_ale)
  296. else
  297.     k = readdir(templates_path .. '*.comp')
  298.     if ((table.getn(k)) == 0) then
  299.         alert("No templates found in:\n" .. templates_path .. ".\n\nScript will use internal ones on the fly :)")
  300.         _cfs['ut'] = 0
  301.     else
  302.         for i = 1, table.getn(k) do
  303.             table.insert(tpl, k[i].Name)
  304.         end
  305.     end
  306. end
  307.  
  308. if ((file_exists(_cfs['Destination'])) == false) then
  309.     _cfs['Destination'] = ''
  310. end
  311.  
  312. if ((file_exists(_cfs['flws'])) == false) then
  313.     _cfs['flws'] = ''
  314. end
  315.  
  316. if ((file_exists(_cfs['sequence'])) == false) then
  317.     _cfs['sequence'] = ''
  318. end
  319. if FusionRemote.RenderManager then
  320.     g   =   FusionRemote.RenderManager:GetGroupList() or {'all'}
  321.     else
  322.     g={'all'}
  323. end
  324.  
  325. opts = {
  326.     {
  327.         "sequence",
  328.         Name = "Scan directory:",
  329.         "PathBrowse",
  330.         Default = (_cfs["sequence"]or ""),
  331.     },{
  332.         "recurse",
  333.         Name = "Recurse? ",
  334.         "Checkbox",
  335.         Default = (_cfs["recurse"]or 1),
  336.         Width = 0.5,
  337.     },{
  338.         "mask",
  339.         Name = "Search for:",
  340.         "Dropdown",
  341.         Options = exts,
  342.         Default = (_cfs["mask"]or 17),
  343.     },{
  344.         "sq",
  345.         Name = 'Convert to:',
  346.         "Dropdown",
  347.         Options = exts2,
  348.         Default = (_cfs["sq"]or 0),
  349.     },{
  350.         "Destination",
  351.         Name = "Save clips to:",
  352.         "PathBrowse",
  353.         Default = (_cfs["Destination"]or ""),
  354.     },{
  355.         "ut",
  356.         Name = "Use templates?",
  357.         "Checkbox",
  358.         NumAcross = 2,
  359.         Default = (_cfs["ut"]or 1),
  360.     },{
  361.         "Comps",
  362.         Name = "Template:",
  363.         "Dropdown",
  364.         Options = tpl,
  365.         Default = (_cfs["Comps"]or ""),
  366.     },{
  367.         "compnames",
  368.         Name = "New clip names:",
  369.         "Dropdown",
  370.         Options ={
  371.             'using sequence name',
  372.             'using folder',
  373.             'using parent folder (1 level up)',
  374.         },
  375.         Default = (_cfs["compnames"]or ""),
  376.     },{
  377.         "jpeg",
  378.         Name = "Make JPG still? ",
  379.         "Checkbox",
  380.         Default = (_cfs["jpeg"]or 1),
  381.         Width = 0.5,
  382.     },{
  383.         "jpegs",
  384.         Name = "JPG Width",
  385.         "Slider",
  386.         Default = (_cfs["jpegs"]or 720),
  387.         Integer = true,
  388.         Min = 120,
  389.         Max = 2000,
  390.         Width = 0.5
  391.     },{
  392.         "counter",
  393.         Name = "Make counter? ",
  394.         "Checkbox",
  395.         Default = (_cfs["counter"]or 1),
  396.         Width = 0.5,
  397.     },{
  398.         "flws",
  399.         Name = "Save generated comps in:",
  400.         "PathBrowse",
  401.         Default = (comp:MapPath(_cfs["flws"]) or ""),
  402.     },{
  403.         "QueueNow",
  404.         Name = "Send to RenderManager?",
  405.         "Checkbox",
  406.         Default = (_cfs["QueueNow"]or 0),
  407.         Width = 0.7,
  408.     },{
  409.         "Submit comps paused",
  410.         Name = 'Paused?',
  411.         "Checkbox",
  412.         Default = (_cfs["Paused"]or 1),
  413.         Width = 0.3,
  414.     },{
  415.         "gs",
  416.         Name = 'Use slaves:',
  417.         "Dropdown",
  418.         Options = g,
  419.     },
  420. }
  421.  
  422. ret = comp:AskUser("Convert footage", opts)
  423. if (not ret) then
  424.     pm('Aborting the script..')
  425.     do
  426.         return
  427.     end
  428. end
  429.  
  430. if (table.getn(g)) then
  431.     groups = g[ret.gs + 1]
  432. else
  433.     groups = "local,all"
  434. end
  435.  
  436. ret.Destination,    ret.sequence,   ret.flws =  addslash(comp:MapPath(ret.Destination)),    addslash(comp:MapPath(ret.sequence)),   addslash(comp:MapPath(ret.flws))
  437. if (ret.sequence == [[\]])  or (ret.flws == [[\]])  or (ret.Destination == [[\]]) then
  438.     alert('\n\n\n\tAborting the script, please provide VALID PATHS.')
  439.     do
  440.         return
  441.     end
  442. elseif ((ret.jpeg == 0) and (exts2[ret.sq + 1]== '- none -')) then
  443.     alert('No output format selected.\n\n (sequence or single jpg)\n\nAborting the script.')
  444.     do
  445.         return
  446.     end
  447. elseif ((file_exists(ret.flws)) == false) then
  448.     alert('Folder for saving new comps does not exist!\n' .. ret.flws)
  449.     do
  450.         return
  451.     end
  452. end
  453. _cfs,   mask = ret, exts[ret.mask + 1]
  454. myscriptprefs('write')
  455. if (fusion:GetPrefs().Global.Network.ServerName == 'localhost') then
  456.     fusion.RenderManager:SetAttrs({ RQUEUEB_Paused = true, })
  457.     warnrm = true
  458.     pm('RenderManager : PAUSED.')
  459. end
  460.  
  461. pm('Searching for '..mask..',please wait...')
  462. if FusionRemote.CacheManager then FusionRemote.CacheManager:Purge() end
  463. local t1 = os.time()
  464. ScanDir(ret.sequence, mask, ret["recurse"])
  465. local cl, timescan = table.getn(clips), os.difftime(os.time(), t1)
  466. local secs = math.fmod(timescan, 60)
  467. local mins = (timescan - secs) / 60
  468. local time1 = mins .. 'm. ' .. secs .. 's. '
  469. if (cl > 0) then
  470.     pm(cl .. ' clips found in ' .. scanned .. ' dirs.\n\tTime: ' .. time1 .. '\nLongest clip has '..maxlen..'\nCreating comps, please wait...')
  471. else
  472.     alert('No ' .. mask .. ' files found in \n' .. ret.sequence .. '\nTime to search:' .. time1)
  473.     do
  474.         return
  475.     end
  476. end
  477. local t2,   user = os.time(),   (os.getenv("USERNAME") or '')
  478. local userinfo = user .. ' at ' .. os.getenv("COMPUTERNAME")
  479. if (ret.ut == 0 or (table.getn(tpl) == 0)) then
  480.     pm('Creating base comp..')
  481.     -- NewComp([boolean quiet[, boolean: autoclose[, boolean: hidden]]])
  482.  
  483.     c = FusionRemote:NewComp(true, true, false) or false
  484.     pos =   getToolPositions(c) or nil
  485.     if not c then do return end end
  486.     SetActiveComp(c)    --SetAttrs({ COMPN_GlobalStart = 1, COMPN_RenderStart = 1 })
  487.     Lock()
  488.     UpdateMode = "All"
  489.     LDR = Loader({ Clip = '' })
  490.     LDR:SetAttrs({TOOLS_Name = "LDR"})
  491.     out = LDR.Output
  492.     if pos then
  493.         c.CurrentFrame.FlowView:SetPos(LDR,0.1,1.1)
  494.     end
  495.     if (ret.counter==1 and (not IsRotation)) then
  496.         Counter=TextPlus{ Center = { 0.9, 0.08}, StyledText=TimeCode{Mins = 0, Hrs = 0, Secs = 0, Flds = 0, Frms = 1}}
  497.         Counter:SetAttrs{ TOOLS_Name = "Counter" }
  498.         MergeCounter = Merge({Background = out, Foreground =Counter.Output})
  499.         MergeCounter:SetAttrs({TOOLS_Name = "MergeCounter"})
  500.         out = MergeCounter.Output
  501.         if pos then
  502.             c.CurrentFrame.FlowView:SetPos(MergeCounter,2,1)
  503.             c.CurrentFrame.FlowView:SetPos(Counter,2,2)
  504.         end
  505.     end
  506.     if (not(exts2[ret.sq + 1]== '- none -')) then
  507.         SQ = Saver({Clip = '', Input = out})
  508.         SQ:SetAttrs({TOOLS_Name = "SQ"})
  509.         if pos then
  510.             c.CurrentFrame.FlowView:SetPos(SQ,4,1)
  511.         end
  512.     end
  513.     if (ret.jpeg == 1) then
  514.         RSZ = BetterResize({Width = ret['jpegs'], Input = out, KeepAspect = 1, Comments = 'Web thumb size.'})
  515.         JPG = Saver({Clip = '', Input = RSZ.Output})
  516.         JPG:SetAttrs({TOOLS_Name = "JPG"})
  517.         if pos then
  518.             c.CurrentFrame.FlowView:SetPos(RSZ,3,2)
  519.             c.CurrentFrame.FlowView:SetPos(JPG,4,2)
  520.         end
  521.     end
  522.     for i, tl in ipairs(c:GetToolList()) do
  523.         tl.TileColor = {R = 1,G = 1,B = 1}
  524.         tl.TextColor = {R = 0,G = 0,B = 0}
  525.         tl.Comments = (tl.Comments or '') .. '\ndon\'t change my name please..'
  526.     end
  527. else
  528.     templatecomp = templates_path .. tpl[ret.Comps + 1]
  529.     pm('Loading template: ' .. templatecomp)
  530.     c = FusionRemote:LoadComp(templatecomp, true,true,true) or false
  531.  
  532.     if (c == false) then
  533.         alert('Unable to load template comp\n\nAborting the script.Please check that file exists and all tools can be loaded.')
  534.         do
  535.             return
  536.         end
  537.     end
  538.  
  539.     if ((not c.JPG) and (not c.SQ)) or (not c.LDR) then
  540.         alert('This comp can not be used as template.\n' .. 'There are missing default tools.\n' .. 'Make sure loader witn name LDR present, and saver with name SQ or JPG')
  541.         do
  542.             return
  543.         end
  544.     end
  545.     SetActiveComp(c)
  546.     c:Lock()
  547.     c.UpdateMode,setldr = "All",    {}
  548.     keep ={
  549.         "Import Mode",
  550.         "Process Mode",
  551.         "Hold First Frame",
  552.         "Hold Last Frame",
  553.         "Reverse",
  554.         "Loop",
  555.         "Missing Frames",
  556.         "Depth",
  557.         "Pixel Aspect",
  558.         "Custom Pixel Aspect",
  559.         "First Frame",
  560.         "Make Alpha solid",
  561.         "Invert Alpha",
  562.         "Post-Multiply by Alpha",
  563.         "Swap Field Dominance",
  564.         "Bypass Conversion",
  565.         "Lock R/G/B",
  566.         "Black Level",
  567.         "White Level",
  568.         "Soft Clip (Knee)",
  569.         "Film Stock Gamma",
  570.         "Conversion Gamma",
  571.         "Green Black Level",
  572.         "Green White Level",
  573.         "Green Soft Clip (Knee)",
  574.         "Green Film Stock Gamma",
  575.         "Green Conversion Gamma",
  576.         "Blue Black Level",
  577.         "Blue White Level",
  578.         "Blue Soft Clip (Knee)",
  579.         "Blue Film Stock Gamma",
  580.         "Blue Conversion Gamma",
  581.         "Conversion Table",
  582.         "Frame Render Script",
  583.         "Start/End Render Scripts",
  584.         "Start Render Script",
  585.         "End Render Script",
  586.         "Comments",
  587.         --"Clip List","Load","Replace","Insert","Import","Format Name","Detect Pulldown Sequence"
  588.         --"KeyCode","Time Code Offset","Video Track","Effect Mask",
  589.         --"Global In","Global Out","Trim In","Trim Out","Apply Mask Inverted",
  590.         --"Multiply by Mask","Blank2","Channel","High","Low","Clip","Proxy Filename",
  591.     }
  592.     for i, input in ipairs(c.LDR:GetInputList()) do
  593.         n = input:GetAttrs().INPS_Name
  594.         if (eyeon.isin(keep, n)) then
  595.             table.insert(setldr, i, {['name']= n,['val']= input[TIME_UNDEFINED]})
  596.         end
  597.     end
  598. end
  599. SetActiveComp(c)
  600. pos =   getToolPositions(c) or nil
  601. pm('Creating comps:')
  602. --if not pos then _ALERT('Unable to fetch tool positions. Expect comps with messed tools :-)') end
  603. cnames = {}
  604. if FusionRemote.CacheManager then FusionRemote.CacheManager:Purge() end
  605. for i, fi in pairs(clips) do --main loop
  606.     SetAttrs({COMPN_GlobalStart = 1, COMPN_GlobalEnd = 500000000000})
  607.     to = (c.LDR.Output:GetConnectedInputs())
  608.     c.LDR:Delete()      --grr...
  609.     LDR = c.Loader({Clip = ''})
  610.     LDR:SetAttrs({TOOLS_Name = "LDR"})
  611.     if pos  then c.CurrentFrame.FlowView:SetPos(LDR,pos['LDR'].x,pos['LDR'].y) end
  612.     for i, x in pairs(to) do    x:ConnectTo(LDR.Output) end
  613.     masp=eyeon.split(fi.Path, [[\]])
  614.     for i, line in ipairs(masp) do
  615.         if (tonumber(string.len(eyeon.trim(line))) > 0) then
  616.             folder = line
  617.             parent= masp[i-1]
  618.         end
  619.     end
  620.     if (ret.compnames == 0) then
  621.         shotname = fi.Name
  622.     elseif(ret.compnames ==1 )  then
  623.         shotname = folder
  624.     else
  625.         shotname = parent
  626.     end
  627.     savedcomp = ret.flws .. shotname .. '.comp'
  628.     if (eyeon.isin(cnames, savedcomp) == true) then  --> check if is uniqie.
  629.         a,j = eyeon.split(fi.FullPath,[[\]]),{}
  630.         a[1],a[table.getn(a)] = nil,nil
  631.         for xc, vv in ipairs(a) do
  632.             table.insert(j, vv)
  633.         end
  634.         o = table.getn(j)
  635.         for i, k in ipairs(j) do
  636.             if (j[o + 1 - i]) then
  637.                 shotname = j[o + 1 - i] .. '.' .. shotname
  638.                 savedcomp = ret.flws .. shotname .. '.comp'
  639.             end
  640.             if (eyeon.isin(cnames, savedcomp) == false) then
  641.                 break
  642.             end
  643.         end
  644.     end
  645.     table.insert(cnames, savedcomp)
  646.     c.LDR.Clip[TIME_UNDEFINED] = fi.FullPath
  647.     wait(0.5)
  648.     if (setldr) then
  649.         for g, input in ipairs(c.LDR:GetInputList()) do
  650.             for o, k in ipairs(setldr) do
  651.                 if (k.name == input:GetAttrs().INPS_Name) then                  --print(n,k.val)
  652.                     input[TIME_UNDEFINED] = k.val
  653.                 end
  654.             end
  655.         end
  656.     end
  657.     c:Save(savedcomp)
  658.     att     = c.LDR:GetAttrs()
  659.     start,  frames = att.TOOLNT_Clip_Start[1],  att.TOOLNT_Clip_End[1]
  660.     if (not frames) then
  661.         alert('UNABLE TO LOAD:\n' .. fi.FullPath)
  662.     else
  663.         c.CurrentTime = (math.ceil(frames / 2))
  664.         if (ret.counter==1 and c.Counter and c.MergeCounter) then
  665.             if (IsRotation) then
  666.                 c.Counter:SetAttrs({TOOLB_PassThrough = true})
  667.                 c.MergeCounter:SetAttrs({TOOLB_PassThrough = true})
  668.             else
  669.                 c.Counter.GlobalOut[TIME_UNDEFINED],c.Counter.Width,c.MergeCounter.Width,c.Counter.Height,c.MergeCounter.Height,
  670.                 c.Counter.Enabled3 = frames,att.TOOLIT_Clip_Width[1],att.TOOLIT_Clip_Width[1],att.TOOLIT_Clip_Height[1],att.TOOLIT_Clip_Height[1],1
  671.             end
  672.         end
  673.         if ((not(exts2[ret.sq + 1]== '- none -')) and c.SQ) then
  674.             c.SQ:SetAttrs({TOOLB_PassThrough = false})
  675.             c.SQ.Clip = ret.Destination .. shotname .. string.lower(exts2[ret.sq + 1])
  676.             --if exts2[ret.sq + 1]=='.omf' and (not IsRotation) then
  677.                 --c.SQ.OMFFormat.Codec[0]='AJPG'
  678.                 --c.SQ.OMFFormat.AJPG.CodecVariety[0]='AVR9s'      -- ADD ANOTHER FILE FORMAT DEFAULT SAVER SETTINGS HERE AS ABOVE FOR OMF
  679.             --end
  680.         else
  681.             if c.SQ then
  682.                 c.SQ:SetAttrs({TOOLB_PassThrough = false})
  683.             end
  684.         end
  685.         -- add another default savers here
  686.  
  687.         if c.OMF and (not IsRotation)then
  688.             if exts2[ret.sq + 1]=='.omf' then
  689.                 c.OMF:SetAttrs({TOOLB_PassThrough = true})
  690.                 pm('Skiping OMF saver, because you have SQ set  to the same OMF!')
  691.             else
  692.                 c.OMF.Clip = ret.Destination .. shotname .. '.omf'
  693.                 --c.OMF.OMFFormat.Codec[0]='AJPG'
  694.                 --c.OMF.OMFFormat.AJPG.CodecVariety[0]='AVR9s'
  695.                 pm('Setting omf saver. Format is'..c.OMF.OMFFormat.AJPG.CodecVariety[0])
  696.             end
  697.         end
  698.  
  699.  
  700.         if c.MOV then
  701.  
  702.             local sq1=eyeon.parseFilename(ret.Destination).Path
  703.             local mov1=eyeon.parseFilename(mov.Clip[TIME_UNDEFINED]).Path
  704.       print('Found custom MOV saver'..mov1)
  705.       print('The default one is '..sq1)
  706.  
  707.             if exts2[ret.sq + 1]=='.mov' then
  708.                 if sq1 == mov1 then
  709.         print('TWO SAVERS IN SAME DIR!! - ADDING _CUSTOM TO YOUR CLIP NAME_')
  710.                             c.MOV.Clip = mov1..'_custom_' .. shotname .. '.mov'
  711.                 end
  712.             else
  713.                 c.MOV.Clip = mov1 .. shotname .. '.mov'
  714.             end
  715.         end
  716.  
  717.     if ((exts2[ret.sq + 1]== '- none -') and (ret.jpeg == 1 and c.JPG)) then -- if creating  only JPG's
  718.             c:SetAttrs({ COMPN_GlobalStart = start,COMPN_GlobalEnd = frames * 2, COMPN_RenderStart = c.CurrentTime,
  719.             COMPN_RenderEnd = c.CurrentTime, COMPN_CurrentTime = c.CurrentTime })
  720.         else
  721.             c:SetAttrs({ COMPN_GlobalStart = start, COMPN_GlobalEnd = frames * 2, COMPN_RenderStart = start,
  722.             COMPN_RenderEnd = frames, COMPN_CurrentTime = c.CurrentTime })
  723.         end
  724.         if (ret.jpeg and c.JPG) and (not IsRotation) then
  725.             c.JPG:SetAttrs({TOOLB_PassThrough = false})
  726.             _j = ret.Destination .. shotname .. '.jpg'
  727.             c.JPG.Clip[0]=_j
  728.             if c.JPG.Blend then
  729.                 c.JPG.Blend =  BezierSpline()
  730.                 c.JPG.Blend[start],c.JPG.Blend[c.CurrentTime - 1],c.JPG.Blend[c.CurrentTime],c.JPG.Blend[c.CurrentTime + 1] = 0,0,1,0
  731.                 if c.JPG.ProcessWhenBlendIs00  then     c.JPG.ProcessWhenBlendIs00[1]=0 end
  732.             end
  733.             c.JPG.FrameSavedScript[0] = '\n_j=' .. '[['.._j..']]' ..
  734.             '\nf=Fusion():MapPath(filename)\n' .. 'if(fileexists(f)) then\n' ..
  735.             'h,e=os.rename(f,_j) \n' .. "if h == true then\n" .. "print(filename..' renamed to '.._j)\n" ..
  736.             "else\n" .. "_ALERT(e,'Error renaming '..filename..' please rename it manualy to '.._j)\n" ..
  737.             "end\n" .. "else\n" .. "_ALERT('Still frame NOT RENDERED !')\n" .. "end\n"
  738.         else
  739.             if c.JPG then c.JPG:SetAttrs({TOOLB_PassThrough = true}) end
  740.         end
  741.         ct = c:GetAttrs()
  742.         c:SetPrefs{
  743.             ["Comp.Info.Comments"] = 'Created using ' .. '\tConvert Footage.eyeonscript by ' ..
  744.             userinfo .. ' on ' .. os.date() .. '\n\nSource:\t' .. fi.FullPath ..
  745.             '\n\nClip:\t[' .. i .. ' from ' .. cl .. ']' .. '\n\nDestination:\t' ..
  746.             ret.Destination .. shotname .. '\n\nComp saved:\t' .. savedcomp,
  747.             ["Comp.Info.ModifyDate"] = os.date()}
  748.         c:Save(savedcomp)
  749.         --eyeon.fprint('Clip ' .. i .. '/' .. cl..':' ,fi.Name,ct.COMPN_RenderStart .. '..' .. ct.COMPN_RenderEnd)
  750.         if (ret["QueueNow"]== 1) and fusion.RenderManager then
  751.             job = fusion:QueueComp({ FileName = savedcomp, Start = start, End = frames, QueuedBy = user, Groups = groups })
  752.             if (ret["Paused"]== 1) then
  753.                 job:SetAttrs{RJOBB_Paused = true}
  754.             end
  755.         end
  756.     end --if frames
  757.     collectgarbage()
  758. end -- clips loop
  759. --c:Unlock()
  760. if ccomp then
  761.     SetActiveComp(ccomp)
  762. end
  763.  
  764.  
  765. while c:IsLocked() == true do
  766. print('unlocking..')
  767.    c:Unlock()
  768. end
  769. --c:Close() --for some reason Fusion 6.0, 6.1 crashes here.
  770.  
  771. local totaltime     =   os.difftime(os.time(), t1)
  772.  
  773. if not FusionRemote.RenderManager then
  774.     _ALERT('Script is finished, but generated comps are not submitted to rendermanager.Unable to connect. Please do this manualy.')
  775. end
  776.  
  777.  
  778.  
  779. endmsg='Script Convert Footage Finished!\nSee console for details.'
  780. --eyeon.fprint('Scaned folders:',scanned ,' in '..time1)
  781. --eyeon.fprint('Clips found:',cl)
  782. --eyeon.fprint('Total time:',((totaltime - math.mod(totaltime, 60)) / 60)..'m. '..(math.mod(totaltime, 60))..'s.')
  783. pm('Comps saved in:\t\t'..ret.flws)
  784.  
  785.  
  786.  
  787. if (cl > 0) then    executebg('explorer /e ,' ..ret.flws)end
  788. if (warnrm) then
  789.     fusion.RenderManager:SetAttrs({ RQUEUEB_Paused = false, })
  790. end
  791. collectgarbage()
  792. alert(endmsg)

I just commented out some fprint statements that weren't working because I couldn't divine what was wrong with them at a glance, and I don't have time to dive deep. Most of the other obvious problems were for statements that needed pairs() or ipairs() calls. Also, math.mod() changed to math.fmod() for some reason. Calls to MapPath() needed to be comp:MapPath(), and there were a couple of places where the script didn't test for the existence of a variable before trying to use it.
Last edited by Midgardsormr on Wed Aug 22, 2018 1:45 pm, edited 1 time in total.

User avatar
RBemendo
Fusioneer
Posts: 153
Joined: Fri Dec 12, 2014 11:32 am
Been thanked: 1 time

Re: Auto applying a template between loader and saver

#18

Post by RBemendo » Wed Aug 22, 2018 12:38 pm

got it, changing the node to something other than IN worked just fine. Thanks.

The other thing I run into is that in one of the .setting files I created, there are multiple loaders, so that's messing up the GetPos(LDR) - I'll have to look at a way to name the specific loader I want to align it to.

User avatar
ShadowMaker SdR
Fusionista
Posts: 655
Joined: Sun Sep 21, 2014 6:17 am
Answers: 4
Been thanked: 17 times

Re: Auto applying a template between loader and saver

#19

Post by ShadowMaker SdR » Wed Aug 22, 2018 1:23 pm

Hi Bryan, it's definitely working better already. I can't believe you managed to get this far in such a short amount of time. It's really appreciated.

Now when I run it, it gets stuck at line 230 with the following message:

...ckmagic Design\Fusion\Scripts\Comp\Convert Footage_2.lua:230: attempt to call a table value

What does that mean?

Also, I don't know exactly what you mean by you don't use Fusion's render manager. I think locally there's always a version of the render manager running so you can make a queue (sp?) of all the comps you're running. I don't think there would be a need to reconfigure your system.

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#20

Post by Midgardsormr » Wed Aug 22, 2018 1:38 pm

That means I missed one of the For statements. Go to that line in the script and where it says for i, tool in c:GetToolList() do, change it to for i, tool in ipairs(c:GetToolList()) do.

In an older version of Lua, used in Fusion 7 and earlier, you could just run a for loop directly on a table. Now you need to use a special iterator function, either pairs() for unordered lists or ipairs() for ordered lists where the order is important. The error message is saying, "Hey, that's a table, but I was expecting a number."

ShadowMaker SdR wrote:
Wed Aug 22, 2018 1:23 pm
Also, I don't know exactly what you mean by you don't use Fusion's render manager. I think locally there's always a version of the render manager running so you can make a queue (sp?) of all the comps you're running. I don't think there would be a need to reconfigure your system.

The script calls the render engine through its network address, but we have Network Rendering disabled for all of our workstations. Thus, the script couldn't reach even the local instance of Fusion.

User avatar
SirEdric
Fusionator
Posts: 1960
Joined: Tue Aug 05, 2014 10:04 am
Answers: 4
Real name: Eric Westphal
Been thanked: 136 times
Contact:

Re: Auto applying a template between loader and saver

#21

Post by SirEdric » Wed Aug 22, 2018 1:40 pm

RBemendo wrote:
Wed Aug 22, 2018 12:38 pm
got it, changing the node to something other than IN worked just fine. Thanks.
"in" and "out" are reserved keywords in Lua.
Although "IN" should be fine, you'll never know...:-)

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#22

Post by Midgardsormr » Wed Aug 22, 2018 1:44 pm

Nope, temporarily activating the Allow Network Renders switch didn't cause it to do anything, either. Updating the script above with the additional for loop fix.

User avatar
RBemendo
Fusioneer
Posts: 153
Joined: Fri Dec 12, 2014 11:32 am
Been thanked: 1 time

Re: Auto applying a template between loader and saver

#23

Post by RBemendo » Wed Sep 26, 2018 5:23 am

I've been putting some more thought to this concept of auto applying settings to comps to help assistants with common tasks, and automating repetitive tasks. A new approach to this would be to create a .setting file that sets up a generic template for common related tasks. So far on my list of repetative greenscreen shots i've come up with this:

-deNoise
-linearize
-Key/Matte
-deSpill
-transform
-pre-grade

-background generation
-lightwrap
-pixel smoosh
-overall grade
-output

The idea I'm thinking about approaching is being able to create .setting files for each step after the generic .setting file gets everything in place. I sometimes work on projects where the same CGI background and grade is used on 30 comps.

The only thing I can think of for "auto placing" nodes in this type of structure is to use the piperouter numbering system to auto connect in the project. Is there another way I should be thinking about this? Ideally I'd eventually get to the point where once the deNoise, linearize, key/matte is completed I run a CMD line script that applies an approved template for the rest of the flow and sends it to the farm for rendering.

A simple look at my thought process:

Code: Select all

{
	Tools = ordered() {
		Loader1 = Loader {
			Clips = {
			},
			Inputs = {
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { -146, 137 } },
		},
		PipeRouter4 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "LoaderFromSaverofDenoise",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 579.799, 114.193 } },
		},
		PipeRouter1 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "PipeRouter3",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 350, 120 } },
		},
		PipeRouter3 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "Gamut1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 254.839, 120.193 } },
		},
		PipeRouter2 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "Loader1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { -24.7117, 136.723 } },
		},
		Note1_3 = Note {
			Inputs = {
				Comments = Input { Value = "Keying template that auto connects to PipeRouter4 and outputs to PipeRouter5", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 788.483, 49.3971 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		Note1_2 = Note {
			Inputs = {
				Comments = Input { Value = "Loader from saver of preComp Denoising ", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 400.099, 10.4079 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		Note1_1 = Note {
			Inputs = {
				Comments = Input { Value = "Linearize", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 42.8783, 66.5852 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		DeltaKeyer1 = DeltaKeyer {
			Inputs = {
				TuningRanges = Input {
					Value = ColorCurves {
						Curves = {
							{
								Points = {
									{ 0, 1 },
									{ 0.4, 0.2 },
									{ 0.6, 0 },
									{ 1, 0 }
								}
							},
							{
								Points = {
									{ 0, 0 },
									{ 0.4, 0 },
									{ 0.6, 0.2 },
									{ 1, 1 }
								}
							}
						}
					},
				},
				Input = Input {
					SourceOp = "PipeRouter4",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 726.858, 260.131 } },
		},
		LoaderFromSaverofDenoise = Loader {
			Clips = {
			},
			NameSet = true,
			Inputs = {
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 468.496, 229.821 } },
		},
		Saver1 = Saver {
			Inputs = {
				ProcessWhenBlendIs00 = Input { Value = 0, },
				Clip = Input {
					Value = Clip {
						Length = 0,
						Saving = true,
						TrimIn = 0,
						ExtendFirst = 0,
						ExtendLast = 0,
						Loop = 1,
						AspectMode = 0,
						Depth = 0,
						GlobalStart = -2000000000,
						GlobalEnd = 0
					},
				},
				OutputFormat = Input { Value = FuID { "TargaFormat" }, },
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
				Input = Input {
					SourceOp = "Denoise",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 462.792, 269.602 } },
		},
		Transform1 = Transform {
			Inputs = {
				Input = Input {
					SourceOp = "PipeRouter5",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 1185.78, 263.923 } },
		},
		MatteControl1 = MatteControl {
			Inputs = {
				Background = Input {
					SourceOp = "DeltaKeyer1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 864.38, 258.712 } },
		},
		Denoise = ChannelBoolean {
			NameSet = true,
			Inputs = {
				Background = Input {
					SourceOp = "PipeRouter1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 338.967, 270.252 } },
		},
		Gamut1 = GamutConvert {
			Inputs = {
				SourceSpace = Input { Value = FuID { "Rec709" }, },
				Input = Input {
					SourceOp = "PipeRouter2",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 89, 256 } },
		},
		PipeRouter5 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "MatteControl1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1012.76, 103.522 } },
		},
		Note1_4 = Note {
			Inputs = {
				Comments = Input { Value = "Position foreground that auto connects to PipeRouter5 and outputs to PipeRouter6\n", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 1142.8, -4.04111 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		ColorCorrector1 = ColorCorrector {
			CtrlWZoom = false,
			Inputs = {
				ColorRanges = Input {
					Value = ColorCurves {
						Curves = {
							{
								Points = {
									{ 0, 1 },
									{ 0.4, 0.2 },
									{ 0.6, 0 },
									{ 1, 0 }
								}
							},
							{
								Points = {
									{ 0, 0 },
									{ 0.4, 0 },
									{ 0.6, 0.2 },
									{ 1, 1 }
								}
							}
						}
					},
				},
				HistogramIgnoreTransparent = Input { Value = 1, },
				Input = Input {
					SourceOp = "PipeRouter6",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 1489.55, 271.467 } },
		},
		PipeRouter6 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "Transform1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1338.06, 102.952 } },
		},
		Note1_5 = Note {
			Inputs = {
				Comments = Input { Value = "Foreground color correction...autoconnects....\n", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 1476.1, 81.1617 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		Merge1 = Merge {
			Inputs = {
				Background = Input {
					SourceOp = "PipeRouter8",
					Source = "Output",
				},
				Foreground = Input {
					SourceOp = "PipeRouter7",
					Source = "Output",
				},
				PerformDepthMerge = Input { Value = 0, },
			},
			ViewInfo = OperatorInfo { Pos = { 1855.11, 104.103 } },
		},
		PipeRouter7 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "ColorCorrector1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1687.5, 106.248 } },
		},
		PipeRouter8 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "ColorCorrector2",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1853.62, 285.276 } },
		},
		lightWrap = ChannelBoolean {
			NameSet = true,
			Inputs = {
				Background = Input {
					SourceOp = "PipeRouter9",
					Source = "Output",
				},
				Foreground = Input {
					SourceOp = "PipeRouter8",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2070.93, 191.49 } },
		},
		PipeRouter10 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "lightWrap",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 2189.41, 99.6766 } },
		},
		PipeRouter9 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "Merge1",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1965.82, 99.6766 } },
		},
		Note1_7 = Note {
			Inputs = {
				Comments = Input { Value = "rest of comp auto connects etc.....", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 2100.22, 14.0462 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		PixelSmoosh = ChannelBoolean {
			NameSet = true,
			Inputs = {
				Background = Input {
					SourceOp = "PipeRouter10",
					Source = "Output",
				},
				Foreground = Input {
					SourceOp = "PipeRouter11",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2310.33, 193.59 } },
		},
		PipeRouter11 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "PipeRouter8",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 2315.62, 322.875 } },
		},
		Note1_6 = Note {
			Inputs = {
				Comments = Input { Value = "background auto connects.....", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 1702.94, 628.157 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		PipeRouter12 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "PixelSmoosh",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 2489.42, 192.193 } },
		},
		ColorCorrector3 = ColorCorrector {
			Inputs = {
				ColorRanges = Input {
					Value = ColorCurves {
						Curves = {
							{
								Points = {
									{ 0, 1 },
									{ 0.4, 0.2 },
									{ 0.6, 0 },
									{ 1, 0 }
								}
							},
							{
								Points = {
									{ 0, 0 },
									{ 0.4, 0 },
									{ 0.6, 0.2 },
									{ 1, 1 }
								}
							}
						}
					},
				},
				HistogramIgnoreTransparent = Input { Value = 1, },
				Input = Input {
					SourceOp = "PipeRouter12",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2609.32, 298.307 } },
		},
		PipeRouter13 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "ColorCorrector3",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 2760.21, 186.804 } },
		},
		Saver2 = Saver {
			Inputs = {
				ProcessWhenBlendIs00 = Input { Value = 0, },
				Clip = Input {
					Value = Clip {
						Length = 0,
						Saving = true,
						TrimIn = 0,
						ExtendFirst = 0,
						ExtendLast = 0,
						Loop = 1,
						AspectMode = 0,
						Depth = 0,
						GlobalStart = -2000000000,
						GlobalEnd = 0
					},
				},
				OutputFormat = Input { Value = FuID { "TargaFormat" }, },
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
				Input = Input {
					SourceOp = "PipeRouter13",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 3014.52, 178.233 } },
		},
		ColorCorrector2 = ColorCorrector {
			Inputs = {
				ColorRanges = Input {
					Value = ColorCurves {
						Curves = {
							{
								Points = {
									{ 0, 1 },
									{ 0.4, 0.2 },
									{ 0.6, 0 },
									{ 1, 0 }
								}
							},
							{
								Points = {
									{ 0, 0 },
									{ 0.4, 0 },
									{ 0.6, 0.2 },
									{ 1, 1 }
								}
							}
						}
					},
				},
				HistogramIgnoreTransparent = Input { Value = 1, },
				Input = Input {
					SourceOp = "Background1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 1841.67, 653.145 } },
		},
		Background1 = Background {
			Inputs = {
				Width = Input { Value = 1920, },
				Height = Input { Value = 1080, },
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 1839.77, 713.903 } },
		}
	}
}

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#24

Post by Midgardsormr » Wed Sep 26, 2018 8:02 am

This is a facsimile of the template comp we use at Muse for our 2d prep task:

Code: Select all

{
	Tools = ordered() {
		ARRILogC = CineonLog {
			NameSet = true,
			Inputs = {
				Depth = Input { Value = 1, },
				LogType = Input { Value = FuID { "ARRILogC" }, },
				SLogVersion = Input { Value = FuID { "SLog2" }, },
				Input = Input {
					SourceOp = "NEAT_Video",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 715, 148.5 } },
		},
		NEAT_Video = BrightnessContrast {
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "SetMetadata1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 385, 148.5 } },
		},
		SetMetadata1 = Fuse.SetMetaData {
			Locked = true,
			Inputs = {
				FieldName = Input { Value = "Creator", },
				FieldValue = Input { Value = "MuseVFX", },
				Input = Input {
					SourceOp = "LDR",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 220, 148.5 } },
		},
		LDR = Loader {
			Clips = {
			},
			NameSet = true,
			Inputs = {
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 110, 148.5 } },
		},
		CineonLog1 = CineonLog {
			Inputs = {
				Depth = Input { Value = 1, },
				SLogVersion = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 715, 181.5 } },
		},
		SetTimeCode1 = Fuse.SetMetaDataTC {
			ViewInfo = OperatorInfo { Pos = { 1210, 214.5 } },
		},
		TimeStretcher1 = TimeStretcher {
			Inputs = {
				SourceTime = Input { Value = 0, },
			},
			ViewInfo = OperatorInfo { Pos = { 990, 214.5 } },
		},
		SLog = CineonLog {
			NameSet = true,
			Inputs = {
				Depth = Input { Value = 1, },
				LogType = Input { Value = FuID { "SonySLog" }, },
				SLogVersion = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 715, 214.5 } },
		},
		Note4 = Note {
			Inputs = {
				Comments = Input { Value = "Denoise. Please don't just use the default settings. Do your best to preserve details by reducing the Luminance amount.", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 330, 214.5 },
				Flags = {
					Expanded = true
				},
				Size = { 178.322, 61.2132 }
			},
		},
		Rec709 = GamutConvert {
			NameSet = true,
			Inputs = {
				SourceSpace = Input { Value = FuID { "Rec709Display" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 715, 247.5 } },
		},
		TimeSpeed1 = TimeSpeed {
			ViewInfo = OperatorInfo { Pos = { 990, 247.5 } },
		},
		sRGB = GamutConvert {
			NameSet = true,
			Inputs = {
				SourceSpace = Input { Value = FuID { "sRGB" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 715, 280.5 } },
		},
		null = BrightnessContrast {
			Locked = true,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "ARRILogC",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 1540, 148.5 } },
		},
		Note1 = Note {
			Inputs = {
				Comments = Input { Value = "Procedures that *do not* change the frame size go here.", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 935, 280.5 },
				Flags = {
					Expanded = true
				},
				Size = { 205.192, 47.7781 }
			},
		},
		Note3 = Note {
			Inputs = {
				Comments = Input { Value = "Linearize your plate.\n\nThe template should be pre-configured for your project, but if there is a question or the conversion looks wrong, consult your supervisor. \n\nIf the conversion is changed from the project default, put the conversion you used in a note in The Grid.", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 660, 313.5 },
				Flags = {
					Expanded = true
				},
				Size = { 196, 179.3 }
			},
		},
		PipeRouter1 = PipeRouter {
			Inputs = {
				Input = Input {
					SourceOp = "null",
					Source = "Output",
				},
			},
			ViewInfo = PipeRouterInfo { Pos = { 1705, 82.5 } },
		},
		LensDistort1 = LensDistort {
			ViewInfo = OperatorInfo { Pos = { 1760, 214.5 } },
		},
		Tracker1 = Tracker {
			Trackers = {
				{
					PatternTime = 378,
					PatternX = 0,
					PatternY = 0
				}
			},
			Inputs = {
				Name1 = Input { Value = "Tracker 1", },
			},
			ViewInfo = OperatorInfo { Pos = { 1760, 247.5 } },
		},
		Note2 = Note {
			Inputs = {
				Comments = Input { Value = "Procedures that change the frame size go here.", }
			},
			ViewInfo = StickyNoteInfo {
				Pos = { 1760, 280.5 },
				Flags = {
					Expanded = true
				},
				Size = { 205.899, 32.2218 }
			},
		},
		Crop1 = Crop {
			Inputs = {
				XSize = Input { Value = 1920, },
				YSize = Input { Value = 1080, },
				Input = Input {
					SourceOp = "LensDistort1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 1870, 214.5 } },
		},
		PlanarTracker1 = Dimension.PlanarTracker {
			Inputs = {
				["Pattern.Polygon"] = Input {
					Value = Polyline {
					},
				},
				Track = Input {
					SourceOp = "PlanarTracker1Track",
					Source = "Value",
				},
				TrackDummy = Input { Value = Matrix { RefTime = 0, IsBogus = true, ToRef = { [0] = 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 } }, },
				StableTrack = Input { Value = Matrix { RefTime = 0, IsBogus = true, ToRef = { [0] = 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 } }, },
			},
			ViewInfo = OperatorInfo { Pos = { 1870, 247.5 } },
		},
		PlanarTracker1Track = BezierSpline {
			SplineColor = { Red = 244, Green = 1, Blue = 129 },
			NameSet = true,
			KeyFrames = {
				[378] = { 0, Flags = { Linear = true, LockedY = true }, Value = Matrix { RefTime = 0, IsBogus = true, ToRef = { [0] = 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1 } } }
			}
		},
		null2 = BrightnessContrast {
			Locked = true,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "null",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2200, 148.5 } },
		},
		Background1 = Background {
			Inputs = {
				Width = Input {
					Value = 1920,
					Expression = "null2.Input.OriginalWidth",
				},
				Height = Input {
					Value = 1080,
					Expression = "null2.Input.OriginalHeight",
				},
				["Gamut.SLogVersion"] = Input { Value = FuID { "SLog2" }, },
			},
			ViewInfo = OperatorInfo { Pos = { 2310, 181.5 } },
		},
		Merge1 = Merge {
			Inputs = {
				Background = Input {
					SourceOp = "null2",
					Source = "Output",
				},
				Foreground = Input {
					SourceOp = "Background1",
					Source = "Output",
				},
				PerformDepthMerge = Input { Value = 0, },
			},
			ViewInfo = OperatorInfo { Pos = { 2310, 148.5 } },
		},
		Gamut1 = GamutConvert {
			Inputs = {
				OutputSpace = Input { Value = FuID { "sRGB" }, },
				Input = Input {
					SourceOp = "Merge1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2475, 214.5 } },
		},
		Scale1 = Scale {
			Inputs = {
				XSize = Input { Value = 0.5, },
				PixelAspect = Input { Value = { 1, 1 }, },
				Input = Input {
					SourceOp = "Gamut1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2640, 247.5 } },
		},
		originalExr = BrightnessContrast {
			Locked = true,
			CtrlWZoom = false,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "PipeRouter1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2750, 82.5 } },
		},
		modifiedExr = BrightnessContrast {
			Locked = true,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "Merge1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2750, 148.5 } },
		},
		fullSizeJpg = BrightnessContrast {
			Locked = true,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "Gamut1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2750, 214.5 } },
		},
		halfSizeJpg = BrightnessContrast {
			Locked = true,
			NameSet = true,
			Inputs = {
				Input = Input {
					SourceOp = "Scale1",
					Source = "Output",
				},
			},
			ViewInfo = OperatorInfo { Pos = { 2750, 247.5 } },
		}
	}
}

My script detects the presence of those locked BrightnessContrast nodes at the right end by name and connects Savers to them. As you can see, all the potentially useful nodes are present in the template, but they're only wired in as needed. Our project management system prompts the user to select a template when starting a new comp, and the templates can be customized per project.

I don't have all the tools we have at work present on this machine at home, so all of those BC nodes would actually be a NEAT Video and a handful of Null Fuses. In any case, the comp creation script loads the template, the user performs the necessary tasks, connecting the pre-set nodes as needed, then runs the script that creates savers, selecting which processes were performed and which outputs need to be created. The script makes the Savers, configures them, and connects them to the appropriate end point nodes. Then the comp is submitted for render.

User avatar
RBemendo
Fusioneer
Posts: 153
Joined: Fri Dec 12, 2014 11:32 am
Been thanked: 1 time

Re: Auto applying a template between loader and saver

#25

Post by RBemendo » Thu Sep 27, 2018 8:22 am

I haven't dug into it too much, but I wonder if I could utilize SirEdric's "ScriptScript" to create the "templates" within the master template comp for a similar type workflow. Your example is really helpful to keep exploring how to modify my workflow.

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#26

Post by Midgardsormr » Thu Sep 27, 2018 10:50 am

You definitely can. Here's an example, my own ToolBox script:

Code: [Select all] [Expand/Collapse] [Download] (MuseToolBox.lua)
  1. DEBUG = false
  2. --[[--
  3.     Fusion MuseToolBox
  4.     v1.0 - 2018-07-03
  5.     by Bryan Ray, inspired by ScriptScript by SirEdric on the We Suck Less forums
  6.  
  7.     =====Overview======
  8.  
  9.     Creates a palette of useful script tools and snippets, organized by use case. Customization of the layout is done
  10.     entirely in the global tables CATEGORIES and BUTTONS.
  11.    
  12.  
  13.     =====License======
  14.  
  15.    
  16.     Change log:
  17.         v1.0: First release version. Refactored for ease of update.
  18.         v0.1: Dev version
  19.  
  20.     To Do:
  21.  
  22.         Detect project and create an optional panel for project-specific tools and scripts
  23.  
  24.         Add custom tools: Macros, Fuses, etc
  25.             Maybe in a tab?
  26.  
  27.  
  28. --]]--
  29.  
  30.  
  31.  
  32.  
  33. -- ===========================================================================
  34. -- constants
  35. -- ===========================================================================
  36.  
  37. SEPARATOR = package.config:sub(1,1) -- Folder separator used by the Operating System.
  38. SCRIPT_PATH = 'S:'..SEPARATOR..'BlackMagic'..SEPARATOR..'fusion9'..SEPARATOR..'scripts'..SEPARATOR
  39.  
  40. -- Set up categories here.
  41. CATEGORIES = {
  42.                 { ID = 'IO', Text = 'I/O Scripts', },
  43.                 { ID = 'compMgmt', Text = 'Comp Management', },
  44.                 { ID = 'utility', Text = 'Utility Scripts',  },
  45.                 { ID = 'ref', Text = 'Scripting Reference', },
  46.                 { ID = 'tools', Text = 'MuseTools',  },
  47.             }
  48.  
  49. MAX_COLUMNS = 3     -- Controls how many buttons can appear in each row
  50.  
  51.  
  52.  
  53.  
  54.  
  55. -- ===========================================================================
  56. -- globals
  57. -- ===========================================================================
  58. _fusion = nil
  59. _comp = nil
  60.  
  61.  
  62.  
  63. -- ===========================================================================
  64. -- Functions                  
  65. -- ===========================================================================
  66.  
  67. function main()
  68.  
  69.     -- get fusion instance
  70.     _fusion = getFusion()
  71.  
  72.  
  73.     -- ensure a fusion instance was retrieved
  74.     if not _fusion then
  75.         error("Please open the Fusion GUI before running this tool.")
  76.     end
  77.  
  78.     -- Set up aliases to the UI Manager framework
  79.     ui = _fusion.UIManager
  80.     disp = bmd.UIDispatcher(ui)
  81.    
  82.  
  83.     -- get composition
  84.     _comp = _fusion.CurrentComp
  85.     SetActiveComp(_comp)
  86.  
  87.  
  88. -- Set up buttons here.
  89.  
  90. -- The script field is a path to a script file. Tool scripts require the tool variable to be sent as an argument in a table. The args field handles this.
  91. -- The macro field is a path to the macro .setting file.
  92. -- The popup field activates a pop-up sub-category window.
  93. -- The tool field holds the TOOLS_RegID value for the tool to be added.
  94.  
  95. BUTTONS = {
  96.     { ID = 'fetchRenders', Text = 'Fetch Renders', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseTools\MT_fetchRenders.lua]], },
  97.     { ID = 'elementBox', Text = 'Element Box', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\z_RnD\ElementBox.py]], },
  98.     { ID = 'twoDPrep', Text = '2d Prep Make Savers', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseTools\2dPrep_MakeSavers.lua]], },
  99.     { ID = 'createSaver', Text = 'Create Matte Saver', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\z_RnD\createSaver_MT.lua]], },
  100.     { ID = 'submitToDeadline', Text = 'Submit to Deadline', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\Submit To Deadline.lua]], },
  101.     { ID = 'versionUp', Text = 'Version Up', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\_Versions\.VersionUp.py]], },
  102.     { ID = 'subversionUp', Text = 'Subversion Up', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\_Versions\.SubversionUp.py]], },
  103.     { ID = 'versionControl', Text = 'Loaders Version Control', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseTools\VersionControl.lua]], },
  104.     { ID = 'batchChangeParams', Text = 'Batch Change Parameters', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\Batch_Change_Parameters.lua]], },
  105.     { ID = 'enableOCL', Text = 'Enable OpenCL All Tools', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\enable_OCL_all_tools.lua]], },
  106.     { ID = 'disableOCL', Text = 'Disable OpenCL All Tools', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\disable_OCL_all_tools.lua]], },
  107.     { ID = 'heatMap', Text = 'Render Time HeatMap', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\thirdparty\hos_HeatMap_Ultra.lua]], },
  108.     { ID = 'highlightAnimated', Text = 'Highlight Animated Nodes', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\reactor\Deploy\Scripts\Comp\EyeonLegacy\HighLight Animated.lua]], },
  109.     { ID = 'fuScriptHelp', Text = 'FusionScript Help Browser', parent = 'ref', script = [[S:\net_exe\BlackMagic\fusion9\reactor\Deploy\Scripts\Comp\UI Manager\FusionScript Help Browser.lua]], },
  110.     { ID = 'actionListener', Text = 'Action Listener', parent = 'ref', script = [[S:\net_exe\BlackMagic\fusion9\reactor\Deploy\Scripts\Comp\UI Manager\Action Listener.lua]], },
  111.     { ID = 'toggleHandles', Text = 'Toggle Handles', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\toggleHandles.lua]], },
  112.     { ID = 'precompThis', Text = 'Precomp This', parent = 'IO', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Tool\Muse_PrecompThis.lua]], args = { tool = _comp.ActiveTool } },
  113.     { ID = 'macro2group', Text = 'Macro to Group', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\Reactor\Deploy\Scripts\Tool\Flow\hos_Macro2Group.lua]], args = { tool = _comp.ActiveTool } },
  114.     { ID = 'alexaFilmGrain', Text = 'Alexa Film Grain', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\AlexaFilmGrain.setting]] },
  115.     { ID = 'flexiTrack', Text = 'FlexiTrack', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_FlexiTrack.setting]] },
  116.     { ID = 'madkey', Text = 'MADKey', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MADKey_BetaD.setting]] },
  117.     { ID = 'lightwrap', Text = 'Fast Lightwrap', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_FastLightWrap.setting]] },
  118.     { ID = 'lensDust', Text = 'Lens Dust', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_LensDust_BETA.setting]] },          
  119.     { ID = 'roughenEdges', Text = 'Roughen Edges', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_RoughenEdges.setting]] },
  120.     { ID = 'softClip', Text = 'Soft Clip Plus', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_SoftClipPlus.setting]] },
  121.     { ID = 'thickener', Text = 'Thickener', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\Thickener.setting]] },
  122.     { ID = 'glitch', Text = 'Glitch Tools', parent = 'tools', popup = 'glitch', args = { path = [[S:\net_exe\BlackMagic\fusion9\macros\Glitch Effects\]] } },
  123.     { ID = 'updateData', Text = 'Update Comp Data', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\updateData.lua]], },
  124.     { ID = 'ToggleTrackerPaths', Text = 'Toggle Tracker Paths', parent = 'utility', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseVFX\ToggleTrackerPaths.lua]], },
  125.     { ID = 'particleSmoke', Text = 'Wispy Smoke', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\settings\MuseVFX\wispySmokeParticles.setting]], },
  126.     { ID = 'heatRipple', Text = 'Heat Ripple', parent = 'tools', macro = [[S:\net_exe\BlackMagic\fusion9\macros\museVFX\MT_HeatRipple.setting]], },
  127.     { ID = 'namespaceManagement', Text = 'Namespace Management', parent = 'compMgmt', script = [[S:\net_exe\BlackMagic\fusion9\scripts\Comp\MuseTools\NamespaceManagement.lua]], },
  128. }
  129.  
  130. --  -- ShotClock
  131. --  -- Save Comp As
  132. --  -- Archive Composition
  133. --  -- SplitEXR_Ultra
  134. --  -- Bake Animation
  135. --  -- ExplodeEXR
  136. --  -- Freeze Camera
  137. -- Glitch Tools
  138. -- Subcategories for MuseTools
  139. -- RSCameraExtractor
  140.  
  141.  
  142.     -- -- ensure a composition is active
  143.     -- if not _comp then
  144.     --  error("Please open a composition before running this tool.")
  145.     -- end
  146.  
  147.     if _comp then dprint("\nActive comp is ".._comp:GetAttrs().COMPS_Name) end
  148.  
  149.     local x = nil
  150.     local y = nil
  151.     local width = 400
  152.     local height = 400
  153.  
  154.     -- Create main window
  155.     window = createMainWindow(x, y, width, height, "MuseToolBox", 'main', '<a href="http://wiki/doku.php?id=fusion#custom_tools">Documentation</a>')
  156.     window:Resize({ width, height })
  157.     content = window:GetItems().root:GetItems().content
  158.  
  159.     -- Collate buttons in each category
  160.     for i, category in ipairs(CATEGORIES) do
  161.         category.buttons = {}
  162.         for i, button in ipairs(BUTTONS) do
  163.             if button.parent == category.ID then
  164.                 table.insert(category.buttons, button)
  165.             end
  166.         end
  167.     end
  168.     ddump(CATEGORIES)
  169.     -- Create category headers
  170.     for i, category in pairs(CATEGORIES) do
  171.         local rows = math.ceil(table.getn(category.buttons) / MAX_COLUMNS)
  172.         local pixelSize = 18
  173.         local gap = 5
  174.         if category.suppressGap == true then
  175.             gap = 0
  176.         end
  177.         if category.suppressLabel == true then
  178.             pixelSize = 1
  179.         end
  180.         content:AddChild(ui:VGroup{ ID = category.ID..'Panel', Weight = 0,
  181.             ui:VGroup{ ID = category.ID..'Header', Weight = 0, ui:Label{ ID = category.ID..'Label', Text = category.Text, Font = ui:Font{PixelSize = pixelSize}, }, },
  182.         })
  183.         for i=1, rows do
  184.             content:GetItems()[category.ID..'Header']:AddChild(ui:HGroup{ID = category.ID..i, Weight = 0, })
  185.         end
  186.         content:GetItems()[category.ID..'Header']:AddChild(ui:VGap(10))
  187.         -- Create buttons
  188.         for i, button in ipairs(category.buttons) do
  189.             local parent = button.parent..math.ceil(i/(table.getn(category.buttons)/rows))
  190.             dprint(parent)
  191.             addUIButton(parent, button.ID, button.Text)
  192.         end
  193.     end
  194.  
  195.  
  196.  
  197.     -- Event handlers
  198.  
  199.     -- close events
  200.     function window.On.main.Close(ev)
  201.         disp:ExitLoop()
  202.         status = "Window closed by user."
  203.     end
  204.  
  205.     -- Create Dynamic Event Handlers
  206.     local dynamicFunctions = {}
  207.     for i, button in ipairs(BUTTONS) do
  208.         dynamicFunctions[button.ID] = {}
  209.         dynamicFunctions[button.ID].Clicked = function (ev)
  210.             if button.macro then
  211.                 _comp:DoAction("AddSetting", { filename = button.macro })
  212.             elseif button.popup then
  213.                 dispatcher(button.popup, button.args)
  214.             else
  215.                 _comp:RunScript(button.script, args)    -- Arguments must be passed as a table.
  216.             end
  217.         end
  218.     end
  219.          
  220.  
  221.     for i, func in pairs(dynamicFunctions) do window.On[i] = func end
  222.  
  223.  
  224.     window:RecalcLayout()
  225.  
  226.     window:Show()
  227.     disp:RunLoop()
  228.     window:Hide()
  229.  
  230.     dprint("Script execution successful. Terminating.")
  231.  
  232. end
  233.  
  234. function dispatcher(module, args)
  235.  
  236.     if module == 'glitch' then
  237.         glitchTools(args.path)
  238.     else
  239.         -- do nothing
  240.     end
  241.  
  242. end
  243.  
  244. function glitchTools(path)
  245.     dprint('glitchTools. Path = '..path)
  246.  
  247.     -- Get a list of Glitch Tools
  248.     local files = listFiles(path, '.setting')
  249.     table.sort(files)
  250.     local rows = math.ceil(table.getn(files) / MAX_COLUMNS)
  251.  
  252.  
  253.     local x = 0
  254.     local y = 0
  255.     local width = 500
  256.     local height = 120 + 35 * rows
  257.     glitchWin = createMainWindow(x, y, width, height, 'Glitch Tools', 'glitch', '<a href="http://wiki/doku.php?id=mee_glitch_tools">Documentation</a>')
  258.  
  259.     content = glitchWin:GetItems().root:GetItems().content
  260.  
  261.     -- Generate HTML for logo header
  262.     html = "<html>\n"
  263.     html = html.."\t<head>\n"
  264.     html = html.."\t\t<style>\n"
  265.     html = html.."\t\t</style>\n"
  266.     html = html .."\t</head>\n"
  267.     html = html .."\t<body>\n"
  268.     html = html .."\t<div>\n"
  269.     html = html .."\t<div style=\"float:right;\">\n"
  270.     html = html .. glitchToolsLogo()
  271.     html = html .. "\t</div>\n"
  272.     html = html .. "\t</body>\n"
  273.     html = html .. "</html>"
  274.  
  275.     glitchWin:GetItems().museLogo.HTML = html
  276.     glitchWin:GetItems().museLogo.MinimumSize = {259, 115}
  277.     glitchWin:GetItems().museLogo.MaximumSize = {width * 2, 115}
  278.     glitchWin:GetItems().museLogo.Alignment = { AlignHCenter = true, AlignTop = true, }
  279.     glitchWin:Resize({ width, height })
  280.  
  281.     -- add rows
  282.     for i=1, rows do
  283.         content:AddChild(ui:HGroup{ ID = 'row'..i, Weight = 0, })
  284.     end
  285.  
  286.     -- Add buttons
  287.     local glitchButtons = {}
  288.     for i, file in ipairs(files) do
  289.         local parent = 'row'..math.ceil(i/(table.getn(files)/rows))
  290.         local text = string.gsub(file, 'MT_', '')
  291.         local text = string.gsub(text, 'Glitch_', '')
  292.         local text = string.gsub(text, '.setting', '')
  293.         addGlitchButton(parent, file, text)
  294.     end
  295.  
  296.     -- Create Dynamic Event Handlers
  297.     local dynamicFunctions = {}
  298.     for i, file in ipairs(files) do
  299.         dynamicFunctions[file] = {}
  300.         dynamicFunctions[file].Clicked = function (ev)
  301.             _comp:DoAction("AddSetting", { filename = path..file })
  302.         end
  303.     end
  304.          
  305.  
  306.     for i, func in pairs(dynamicFunctions) do glitchWin.On[i] = func end
  307.  
  308.     -- close events
  309.     function glitchWin.On.glitch.Close(ev)
  310.         glitchWin:Hide()
  311.         status = "Window closed by user."
  312.     end
  313.  
  314.     glitchWin:Show()
  315.  
  316. end
  317.  
  318. function addGlitchButton(parent, ID, name)
  319.     if not name then
  320.         name = ID
  321.     end
  322.  
  323.     local content = glitchWin:GetItems().root:GetItems()[parent]
  324.     content:AddChild(ui:Button{ ID = ID, Weight = 1, Text = name, })
  325. end
  326.  
  327. function addUIButton(parent, ID, name)
  328. --  dprint(parent..", "..ID..", "..name)
  329.     if not name then
  330.         name = ID
  331.     end
  332.  
  333.     local content = window:GetItems().root:GetItems()[parent]
  334.     content:AddChild(ui:Button{ ID = ID, Weight = 1, Text = name, })
  335. end
  336.  
  337.  
  338.  
  339. ------------------------------------------------------------------------
  340. -- getFusion()
  341. --
  342. -- check if global fusion is set, meaning this script is being
  343. -- executed from within fusion
  344. --
  345. -- Arguments: None
  346. -- Returns: handle to the Fusion instance
  347. ------------------------------------------------------------------------
  348. function getFusion()
  349.     if fusion == nil then
  350.         -- remotely get the fusion ui instance
  351.         fusion = bmd.scriptapp("Fusion", "localhost")
  352.     end
  353.     return fusion
  354. end -- end of getFusion()
  355.  
  356.  
  357.  
  358. -------------------------------------------------------------------------
  359. -- listFiles(path)
  360. --
  361. -- Returns a table containing a sorted list of files with the designated
  362. -- extension
  363. --
  364. -- Arguments:
  365. --      path, string
  366. --      extension, string
  367. -- Returns:
  368. --      fileList, table, a list of files found at the end of the path
  369. -------------------------------------------------------------------------
  370. function listFiles(path, extension)
  371.     dprint('listFiles. Path = '..path..', Extension: '..extension)
  372.  
  373.     local fileList = {}
  374.     local data = bmd.readdir(path..'*'..extension)
  375.     for i, file in ipairs(data) do
  376.         table.insert(fileList, file.Name)
  377.     end
  378.     table.sort(fileList)
  379.     return fileList
  380. end -- end of listFolders(path)
  381.  
  382.  
  383.  
  384. --Functions lifted from bmd.scriptlib
  385. -----------------------------------------------------
  386. -- split(strInput, delimit)
  387. --
  388. -- converts string strInput into a table, separating
  389. -- records using the provided delimiter string
  390. --
  391. -----------------------------------------------------
  392. function split(strInput, delimit)
  393.     local strLength
  394.     local strTemp
  395.     local strCollect
  396.     local tblSplit
  397.     local intCount
  398.  
  399.     tblSplit = {}
  400.     intCount = 0
  401.     strCollect = ""
  402.     if delimit == nil then
  403.         delimit = ","
  404.     end
  405.  
  406.     strLength = string.len(strInput)
  407.     for i = 1, strLength do
  408.         strTemp = string.sub(strInput, i, i)
  409.         if strTemp == delimit then
  410.             intCount = intCount + 1
  411.             tblSplit[intCount] = trim(strCollect)
  412.             strCollect = ""
  413.         else
  414.             strCollect = strCollect .. strTemp
  415.         end
  416.     end
  417.     intCount = intCount + 1
  418.     tblSplit[intCount] = trim(strCollect)
  419.  
  420.     return tblSplit
  421. end
  422.  
  423. -----------------------------------------------------
  424. -- trim(strTrim)
  425. --
  426. -- returns strTrim with leading and trailing spaces
  427. -- removed.
  428. --
  429. -- introduced bmd.dfscriptlib v1.0
  430. -- last updated in v1.3
  431. -----------------------------------------------------
  432. function trim(strTrim)
  433.     strTrim = string.gsub(strTrim, "^(%s+)", "") -- remove leading spaces
  434.     strTrim = string.gsub(strTrim, "(%s+)$", "") -- remove trailing spaces
  435.     return strTrim
  436. end
  437.  
  438.  
  439. ------------------------------------------------------------------------------
  440. -- parseFilename()
  441. --
  442. -- this is a great function for ripping a filepath into little bits
  443. -- returns a table with the following
  444. --
  445. -- FullPath : The raw, original path sent to the function
  446. -- Path     : The path, without filename
  447. -- FullName : The name of the clip w\ extension
  448. -- Name     : The name without extension
  449. -- CleanName: The name of the clip, without extension or sequence
  450. -- SNum     : The original sequence string, or "" if no sequence
  451. -- Number   : The sequence as a numeric value, or nil if no sequence
  452. -- Extension: The raw extension of the clip
  453. -- Padding  : Amount of padding in the sequence, or nil if no sequence
  454. -- UNC      : A true or false value indicating whether the path is a UNC path or not
  455. ------------------------------------------------------------------------------
  456. function bmd.parseFilename(filename)
  457.     local seq = {}
  458.     seq.FullPath = filename
  459.     string.gsub(seq.FullPath, "^(.+[/\\])(.+)", function(path, name) seq.Path = path seq.FullName = name end)
  460.     string.gsub(seq.FullName, "^(.+)(%..+)$", function(name, ext) seq.Name = name seq.Extension = ext end)
  461.  
  462.     if not seq.Name then -- no extension?
  463.         seq.Name = seq.FullName
  464.     end
  465.  
  466.     string.gsub(seq.Name,     "^(.-)(%d+)$", function(name, SNum) seq.CleanName = name seq.SNum = SNum end)
  467.  
  468.     if seq.SNum then
  469.         seq.Number = tonumber( seq.SNum )
  470.         seq.Padding = string.len( seq.SNum )
  471.     else
  472.        seq.SNum = ""
  473.        seq.CleanName = seq.Name
  474.     end
  475.  
  476.     if seq.Extension == nil then seq.Extension = "" end
  477.     seq.UNC = ( string.sub(seq.Path, 1, 2) == [[\\]] )
  478.  
  479.     return seq
  480. end
  481.  
  482. --==================================================================================================================
  483. -- UI MANAGER
  484. --==================================================================================================================
  485.  
  486. --=================================================================
  487. -- createMainWindow(x, y, width, height, title)
  488. --
  489. -- Creates a generic UI Manager window featuring the Muse VFX
  490. -- Logo header. There is an empty content container into
  491. -- which other functions can load information.
  492. --
  493. -- Arguments:
  494. --      x, y: Integer. Screen coordinates where window will appear
  495. --      width, height: Integer. Window size in pixels
  496. --      title: String. Window title.
  497. -- Returns:
  498. --      window: UIWindow object
  499. --=================================================================
  500. function createMainWindow(x, y, width, height, title, id, link)
  501.     dprint('createMainWindow')
  502.  
  503.     -- Generate HTML for logo header
  504.     html = "<html>\n"
  505.     html = html.."\t<head>\n"
  506.     html = html.."\t\t<style>\n"
  507.     html = html.."\t\t</style>\n"
  508.     html = html .."\t</head>\n"
  509.     html = html .."\t<body>\n"
  510.     html = html .."\t<div>\n"
  511.     html = html .."\t<div style=\"float:right;\">\n"
  512.     html = html .. museLogo()
  513.     html = html .. "\t</div>\n"
  514.     html = html .. "\t</body>\n"
  515.     html = html .. "</html>"
  516.  
  517.     -- Create the window
  518.     local window = disp:AddWindow({
  519.  
  520.         -- Window properties
  521.         ID = id,
  522.         WindowTitle = title,
  523. --      Geometry = {x, y, width, height},
  524.  
  525.    
  526.         -- Main window container  
  527.         ui:VGroup{
  528.             ID = 'root',
  529.  
  530.             -- The logo header.
  531.             ui:VGroup{
  532.                 ID = 'header',
  533.                 Weight = 0,
  534.                 ui:TextEdit{ ID = 'museLogo', ReadOnly = true, Alignment = { AlignHCenter = true, AlignTop = true, }, MinimumSize = {286, 80}, MaximumSize = {width * 2, 80}, HTML = html, },
  535.             },
  536.            
  537.             -- This holds the dynamic content
  538.             ui:VGroup{
  539.                 ID = 'content',
  540.                 Weight = 2.0,
  541.             },
  542.  
  543.             -- Footer contains the window control buttons
  544.             ui:VGroup{
  545.                 ID = 'footer',
  546.                 Weight = 0.0,
  547.                 ui:HGroup{
  548.                     ID = 'control',
  549.                     Weight = 0.0,
  550.  
  551.                     ui:Label{
  552.                         ID = 'link',
  553.                         Text = link,
  554.                         Alignment = {AlignRight = true, AlignTop = true, },
  555.                         WordWrap = true,
  556.                         OpenExternalLinks = true,
  557.                     },
  558.  
  559.  
  560.        --           ui:HGap(width - 300),
  561.  
  562.        --           ui:Button{
  563.        --               ID = 'cancel',
  564.        --               Text = 'Cancel',
  565.        --           },
  566.  
  567.           --        ui:Button{
  568.           --            ID = 'next',
  569.           --            Text = 'Next',
  570.           --        },
  571.                 },
  572.             },
  573.         },
  574.     })
  575.  
  576.     return window
  577. end
  578.  
  579.  
  580. --================================================
  581. -- Muse VFX Logo encoded in Base 64 for UI windows
  582. --================================================
  583. function museLogo()
  584.     return [[<img src=''/>]]
  585. end
  586.  
  587. -- Glitch Tools Logo
  588. function glitchToolsLogo()
  589.     return [[<img src = ''/>]]
  590. end
  591.  
  592.  
  593. function removeUIContent(window)
  594.     dprint("Removing widgets from UI: ", true)
  595.     for i, item in pairs(window:GetItems().root:GetItems().content:GetItems()) do
  596.         window:GetItems().root:GetItems().content:RemoveChild(item.ID)
  597.         dprint(item.ID..", ", true)
  598.     end
  599.     dprint('')
  600. end
  601.  
  602.  
  603.  
  604. --========================== COMP MANIPULATION ============================--
  605.  
  606. ----------------------------------------------------------------------------------------------
  607. -- Fusion can be locked or unlocked multiple times. These functions ensure that the lock state
  608. -- is not nested.
  609. ----------------------------------------------------------------------------------------------
  610.  
  611. -- Unlocks the comp
  612. function unlockComp(c)
  613.     if c:GetAttrs().COMPB_Locked == false then lockComp(c) end
  614.     while c:GetAttrs().COMPB_Locked == true do
  615.         c:Unlock()
  616.     end
  617. end
  618.  
  619. -- Locks the comp
  620. function lockComp(c)
  621.     if c:GetAttrs().COMPB_Locked == true then unlockComp(c) end
  622.     while c:GetAttrs().COMPB_Locked == false do
  623.         c:Lock()
  624.     end
  625. end
  626.  
  627.  
  628. --========================== DEBUGGING ============================--
  629.  
  630.  
  631. ---------------------------------------------------------------------
  632. -- dprint(string, suppressNewline)
  633. --
  634. -- Prints debugging information to the console when DEBUG flag is set
  635. --
  636. -- Arguments:
  637. --      string, string, a message for the console
  638. --      suppressNewline, boolean, do not start a new line
  639. ---------------------------------------------------------------------
  640. function dprint(string, suppressNewline)
  641.     local newline = "\n"
  642.     if suppressNewline then newline = '' end
  643.     if DEBUG then _comp:Print(string..newline) end
  644. end -- dprint()
  645.  
  646. ---------------------------------------------------------------------
  647. -- ddump(object)
  648. --
  649. -- Performs a dump() if the DEBUG flag is set
  650. --
  651. -- Arguments
  652. --      object, object, an object to be dumped
  653. ---------------------------------------------------------------------
  654. function ddump(object)
  655.     if DEBUG then dump(object) end
  656. end -- end ddump()
  657.  
  658.  
  659. main()

Obviously none of the buttons will actually do anything outside of our environment, but it should be reasonably easy to update it to work elsewhere.

Actually, I should do a case study of this thing and put a generic version in Reactor. I'm quite proud of it.

User avatar
SirEdric
Fusionator
Posts: 1960
Joined: Tue Aug 05, 2014 10:04 am
Answers: 4
Real name: Eric Westphal
Been thanked: 136 times
Contact:

Re: Auto applying a template between loader and saver

#27

Post by SirEdric » Thu Sep 27, 2018 12:44 pm

Midgardsormr wrote:
Thu Sep 27, 2018 10:50 am
by Bryan Ray, inspired by ScriptScript by SirEdric on the We Suck Less forums
Thanks, mate...:-)
Also, nice to see dprint() being used elsewhere.
Glad to be part of the igniting spark.

Cheers.

Eric.

User avatar
RBemendo
Fusioneer
Posts: 153
Joined: Fri Dec 12, 2014 11:32 am
Been thanked: 1 time

Re: Auto applying a template between loader and saver

#28

Post by RBemendo » Thu Sep 27, 2018 2:01 pm

Brian, WOW!

That's exactly what I had in mind when thinking about how to approach this, simply amazing!

A case study would be fantastic, it's going to take a me a lifetime to figure out how you did all of that.

User avatar
Midgardsormr
Fusionator
Posts: 1215
Joined: Wed Nov 26, 2014 8:04 pm
Answers: 3
Location: Los Angeles, CA, USA
Been thanked: 95 times
Contact:

Re: Auto applying a template between loader and saver

#29

Post by Midgardsormr » Tue May 21, 2019 3:18 pm

Ran across this thread again while hunting for something else and realized I'd never linked the case study:
http://www.bryanray.name/wordpress/tool ... or-fusion/

I haven't generalized it for inclusion in Reactor yet, though.


User avatar
RBemendo
Fusioneer
Posts: 153
Joined: Fri Dec 12, 2014 11:32 am
Been thanked: 1 time

Re: Auto applying a template between loader and saver

#30

Post by RBemendo » Thu May 30, 2019 6:02 am

This is fantastic Bryan! Thanks for updating this here, I certainly missed it.


I just ran into something with the "auto apply" template where I'd like to update a template i've already applied to .comps. Is there a way to Delete all nodes between LDR and SVR, then continue the script for auto applying?