Blame IlmImf/ImfPartHelper.h

Packit Service 6754ca
///////////////////////////////////////////////////////////////////////////
Packit Service 6754ca
//
Packit Service 6754ca
// Copyright (c) 2012, Weta Digital Ltd
Packit Service 6754ca
// 
Packit Service 6754ca
// All rights reserved.
Packit Service 6754ca
// 
Packit Service 6754ca
// Redistribution and use in source and binary forms, with or without
Packit Service 6754ca
// modification, are permitted provided that the following conditions are
Packit Service 6754ca
// met:
Packit Service 6754ca
// *       Redistributions of source code must retain the above copyright
Packit Service 6754ca
// notice, this list of conditions and the following disclaimer.
Packit Service 6754ca
// *       Redistributions in binary form must reproduce the above
Packit Service 6754ca
// copyright notice, this list of conditions and the following disclaimer
Packit Service 6754ca
// in the documentation and/or other materials provided with the
Packit Service 6754ca
// distribution.
Packit Service 6754ca
// *       Neither the name of Weta Digital nor the names of
Packit Service 6754ca
// its contributors may be used to endorse or promote products derived
Packit Service 6754ca
// from this software without specific prior written permission. 
Packit Service 6754ca
// 
Packit Service 6754ca
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Packit Service 6754ca
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Packit Service 6754ca
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
Packit Service 6754ca
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
Packit Service 6754ca
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
Packit Service 6754ca
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
Packit Service 6754ca
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
Packit Service 6754ca
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
Packit Service 6754ca
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
Packit Service 6754ca
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
Packit Service 6754ca
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
Packit Service 6754ca
//
Packit Service 6754ca
///////////////////////////////////////////////////////////////////////////
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
#ifndef INCLUDED_IMF_PARTHELPER_H
Packit Service 6754ca
#define INCLUDED_IMF_PARTHELPER_H
Packit Service 6754ca
Packit Service 6754ca
//-----------------------------------------------------------------------------
Packit Service 6754ca
//
Packit Service 6754ca
//	Functions to help split channels into separate parts: provide a list of
Packit Service 6754ca
//      channels, with desired views. call SplitChannels to assign a part to each
Packit Service 6754ca
//      layer, or correct the name of the channel.
Packit Service 6754ca
//      Also can enumerate the parts in a file and list which parts channels are in
Packit Service 6754ca
//
Packit Service 6754ca
//      This is a good way to offer a 'create Multipart file' checkbox to the user in a
Packit Service 6754ca
//      write dialog box: Populate a list of MultiViewChannelName objects,
Packit Service 6754ca
//      call SplitChannels with whether single or multipart files are required.
Packit Service 6754ca
//      Then write the number of parts it specifies, using internal_name for the channel
Packit Service 6754ca
//      names in the ChannelList and FrameBuffer objects. There should be no need
Packit Service 6754ca
//      for different codepaths for single part and multipart files
Packit Service 6754ca
//
Packit Service 6754ca
//      Similarly, on reading a file as a MultiPartInputFile, use GetChannelsInMultiPartFile to
Packit Service 6754ca
//      enumerate all channels in the file, using internal_name in FrameBuffer objects
Packit Service 6754ca
//      to read the channel
Packit Service 6754ca
//   
Packit Service 6754ca
//
Packit Service 6754ca
//-----------------------------------------------------------------------------
Packit Service 6754ca
Packit Service 6754ca
#include "ImfForward.h"
Packit Service 6754ca
#include "ImfNamespace.h"
Packit Service 6754ca
#include "ImfExport.h"
Packit Service 6754ca
#include "ImfMultiPartInputFile.h"
Packit Service 6754ca
#include "ImfChannelList.h"
Packit Service 6754ca
#include "ImfStringVectorAttribute.h"
Packit Service 6754ca
#include "ImfStandardAttributes.h"
Packit Service 6754ca
#include "ImfMultiView.h"
Packit Service 6754ca
Packit Service 6754ca
#include <string>
Packit Service 6754ca
#include <map>
Packit Service 6754ca
#include <set>
Packit Service 6754ca
Packit Service 6754ca
OPENEXR_IMF_INTERNAL_NAMESPACE_HEADER_ENTER
Packit Service 6754ca
Packit Service 6754ca
struct MultiViewChannelName{
Packit Service 6754ca
  
Packit Service 6754ca
public:
Packit Service 6754ca
  std::string name;         ///< name of channel
Packit Service 6754ca
  std::string view;         ///< view for channel
Packit Service 6754ca
  
Packit Service 6754ca
  int part_number;          ///< part number: updated by SplitChannels
Packit Service 6754ca
  std::string internal_name;///< name used in headers: in singlepart mode, may contain viewname
Packit Service 6754ca
  
Packit Service 6754ca
  virtual ~MultiViewChannelName() {}
Packit Service 6754ca
    
Packit Service 6754ca
    //return layer for this channel, or "" if no layer
Packit Service 6754ca
    std::string getLayer() const 
Packit Service 6754ca
    {
Packit Service 6754ca
        std::size_t q=name.rfind('.');
Packit Service 6754ca
	if(  q==name.npos  )
Packit Service 6754ca
	{
Packit Service 6754ca
	    return "";
Packit Service 6754ca
	}
Packit Service 6754ca
	return name.substr(0,q);
Packit Service 6754ca
    
Packit Service 6754ca
  }
Packit Service 6754ca
Packit Service 6754ca
  std::string getSuffix() const
Packit Service 6754ca
  {
Packit Service 6754ca
        std::size_t q=name.rfind('.');
Packit Service 6754ca
	if(  q==name.npos  )
Packit Service 6754ca
	{
Packit Service 6754ca
	    return name;
Packit Service 6754ca
	}
Packit Service 6754ca
	return name.substr(q+1);
Packit Service 6754ca
      
Packit Service 6754ca
  }
Packit Service 6754ca
  
Packit Service 6754ca
};
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
Packit Service 6754ca
//
Packit Service 6754ca
///\brief assigns individual channels to different parts based on their layer and view name
Packit Service 6754ca
///       input is an array, list, vector etc of MultiViewChannelName objects
Packit Service 6754ca
///       on entry, each MultiViewChannelName name/view must be set (view can be empty if not multiview)
Packit Service 6754ca
///
Packit Service 6754ca
///       if singlepart set, then on exit part_number will be zero, and internal_name will have view name inserted
Packit Service 6754ca
///       otherwise, each channel will be assigned to a different part based on its layer name and view name
Packit Service 6754ca
///
Packit Service 6754ca
/// @param begin pointer to first MultiViewChannelName item
Packit Service 6754ca
/// @param end   pointer to end of MultiViewChannelName item array
Packit Service 6754ca
/// @return      total number of parts required
Packit Service 6754ca
//
Packit Service 6754ca
Packit Service 6754ca
template<typename T> int 
Packit Service 6754ca
SplitChannels(const T & begin,const T & end,bool multipart=true,const std::string & heroView="")
Packit Service 6754ca
{
Packit Service 6754ca
    if(!multipart)
Packit Service 6754ca
    {
Packit Service 6754ca
	for(T i=begin;i!=end;i++)
Packit Service 6754ca
	{
Packit Service 6754ca
	    i->part_number=0;
Packit Service 6754ca
	
Packit Service 6754ca
	    //does this have a view name set?
Packit Service 6754ca
	    if(i->view=="")
Packit Service 6754ca
	    {
Packit Service 6754ca
		i->internal_name=i->name;
Packit Service 6754ca
	    }else{
Packit Service 6754ca
		
Packit Service 6754ca
		std::string lname = i->getLayer();
Packit Service 6754ca
	   
Packit Service 6754ca
		// no layer, only non-hero views get view name in layer name
Packit Service 6754ca
	
Packit Service 6754ca
	    
Packit Service 6754ca
		if(lname=="")
Packit Service 6754ca
		{
Packit Service 6754ca
		    if(i->view==heroView)
Packit Service 6754ca
		    {
Packit Service 6754ca
			i->internal_name = i->name;
Packit Service 6754ca
		    }else{
Packit Service 6754ca
			i->internal_name = i->view+"."+i->name;
Packit Service 6754ca
		    }
Packit Service 6754ca
		}else{
Packit Service 6754ca
		    i->internal_name = lname+"."+i->view+"."+i->getSuffix();
Packit Service 6754ca
		}
Packit Service 6754ca
	    }
Packit Service 6754ca
	}
Packit Service 6754ca
	// single part created
Packit Service 6754ca
	return 1;
Packit Service 6754ca
    }else{
Packit Service 6754ca
	// step 1: extract individual layers and parts
Packit Service 6754ca
	// for each layer, enumerate which views are active
Packit Service 6754ca
	
Packit Service 6754ca
	std::map< std::string , std::set< std::string > > viewsInLayers;
Packit Service 6754ca
	for(T i=begin;i!=end;i++)
Packit Service 6754ca
	{
Packit Service 6754ca
	    viewsInLayers[i->getLayer()].insert(i->view);
Packit Service 6754ca
	}
Packit Service 6754ca
	
Packit Service 6754ca
	// step 2: assign a part number to each layer/view
Packit Service 6754ca
	
Packit Service 6754ca
	std::map< std::pair<std::string,std::string> , int > layerToPart;
Packit Service 6754ca
	
Packit Service 6754ca
	int partCount=0;
Packit Service 6754ca
	
Packit Service 6754ca
	for(std::map< std::string , std::set< std::string > >::const_iterator layer=viewsInLayers.begin();
Packit Service 6754ca
	    layer!=viewsInLayers.end();layer++)
Packit Service 6754ca
	{
Packit Service 6754ca
	    // if this layer has a heroView, insert that first
Packit Service 6754ca
	    bool layer_has_hero = layer->second.find(heroView)!=layer->second.end();
Packit Service 6754ca
	    if( layer_has_hero )
Packit Service 6754ca
	    {
Packit Service 6754ca
		layerToPart[ std::make_pair(layer->first,heroView) ] = partCount++;
Packit Service 6754ca
	    }
Packit Service 6754ca
	    
Packit Service 6754ca
	    
Packit Service 6754ca
	    // insert other layers which aren't the hero view
Packit Service 6754ca
	    for(std::set< std::string >::const_iterator view=layer->second.begin();
Packit Service 6754ca
		view!=layer->second.end();view++)
Packit Service 6754ca
	    {
Packit Service 6754ca
		if(*view!=heroView)
Packit Service 6754ca
		{
Packit Service 6754ca
		    layerToPart[ std::make_pair(layer->first,*view) ] = partCount++;
Packit Service 6754ca
		}
Packit Service 6754ca
	    }
Packit Service 6754ca
		
Packit Service 6754ca
	}
Packit Service 6754ca
	
Packit Service 6754ca
	// step 3: update part number of each provided channel
Packit Service 6754ca
	
Packit Service 6754ca
	for( T i=begin;i!=end;i++)
Packit Service 6754ca
	{
Packit Service 6754ca
	    i->internal_name=i->name;
Packit Service 6754ca
	    i->part_number = layerToPart[ std::make_pair(i->getLayer(),i->view) ];
Packit Service 6754ca
	}
Packit Service 6754ca
	
Packit Service 6754ca
	
Packit Service 6754ca
	// return number of parts created
Packit Service 6754ca
	return partCount;
Packit Service 6754ca
    }
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
//
Packit Service 6754ca
// populate the chans vector<MultiViewChannelName> with a list of channels in the file
Packit Service 6754ca
// and their corresponding part number
Packit Service 6754ca
//
Packit Service 6754ca
template<class T> void
Packit Service 6754ca
GetChannelsInMultiPartFile(const MultiPartInputFile & file,T & chans)
Packit Service 6754ca
{
Packit Service 6754ca
    bool has_multiview=false;
Packit Service 6754ca
    StringVector mview; 
Packit Service 6754ca
    if(file.parts()==1)
Packit Service 6754ca
    {
Packit Service 6754ca
	if(hasMultiView(file.header(0)))
Packit Service 6754ca
	{
Packit Service 6754ca
	    mview=multiView(file.header(0));
Packit Service 6754ca
	    has_multiview=true;
Packit Service 6754ca
	}
Packit Service 6754ca
    }
Packit Service 6754ca
    
Packit Service 6754ca
    for(int p=0;p
Packit Service 6754ca
    {
Packit Service 6754ca
	const ChannelList & c=file.header(p).channels();
Packit Service 6754ca
	
Packit Service 6754ca
	std::string view="";
Packit Service 6754ca
	if(file.header(p).hasView())
Packit Service 6754ca
	{
Packit Service 6754ca
	    view=file.header(p).view();
Packit Service 6754ca
	}
Packit Service 6754ca
	for(ChannelList::ConstIterator i=c.begin();i!=c.end();i++)
Packit Service 6754ca
	{
Packit Service 6754ca
	    MultiViewChannelName m;
Packit Service 6754ca
            m.name=std::string(i.name());
Packit Service 6754ca
	    m.internal_name=m.name;
Packit Service 6754ca
Packit Service 6754ca
	    if(has_multiview)
Packit Service 6754ca
	    {
Packit Service 6754ca
		m.view=viewFromChannelName(m.name,mview);
Packit Service 6754ca
		m.name=removeViewName(m.internal_name,m.view);
Packit Service 6754ca
	    }else{
Packit Service 6754ca
		m.view=view;
Packit Service 6754ca
	    }
Packit Service 6754ca
            m.part_number=p;
Packit Service 6754ca
	    chans.push_back(m);
Packit Service 6754ca
		
Packit Service 6754ca
	}
Packit Service 6754ca
    }
Packit Service 6754ca
}
Packit Service 6754ca
Packit Service 6754ca
OPENEXR_IMF_INTERNAL_NAMESPACE_HEADER_EXIT
Packit Service 6754ca
Packit Service 6754ca
#endif