Author Topic: Unable to copy and resample image  (Read 8300 times)

Offline David Serrano

  • PTeam Member
  • PixInsight Guru
  • ****
  • Posts: 503
Unable to copy and resample image
« on: 2007 September 17 01:38:43 »
Hi,

Consider the following code:

Code: [Select]
#include <pjsr/Interpolation.jsh>
#include <pjsr/SampleType.jsh>
#include <pjsr/UndoFlag.jsh>

var w = ImageWindow.activeWindow;
if (w.isNull)
    throw Error ("No image window.");

var v = w.mainView;

var copy_img = new ImageWindow (
    v.image.width, v.image.height,
    v.image.numberOfChannels, v.image.bitsPerSample,
    v.image.sampleType == SampleType_Real,
    true, "foobar23204"
);

with (copy_img.mainView) {
    beginProcess (UndoFlag_NoSwapFile);
    //image.apply (v.image);
    //image.assign (v.image);
    //image.clonePixelData (v.image);
    //var ap = new Array; v.image.getPixels (ap); image.setPixels (ap);
    //image.transfer (v.image);
    endProcess();
    image.resample (640, 480, Interpolation_BicubicBSpline);
}

copy_img = null;
gc();


This code creates a new ImageWindow and tries to copy the data from another image. As you can see, I've tried several ways ;) to no avail. I've also tried commenting out the beginProcess/endProcess lines because I have little idea of what that methods do. Didn't found a method called .copy, .duplicate or so.

It's worth noting that the code as shown, without modifications (ie, without uncommenting one of the 5 methods I tried to duplicate the image), doesn't work yet. It says "read only image" when resampling. This is the error that appears in most of the 10 combinations I tried (if not all), albeit in different lines sometimes.

I've also tried (from memory):

Code: [Select]
var copy2 = new ImageWindow (w);   // duplicate window
copy2.mainView.image.resample (640, 480, blah_blah);


and got a similar "read only" error.

Some input on this would be appreciated ;).
--
 David Serrano

Offline Juan Conejero

  • PTeam Member
  • PixInsight Jedi Grand Master
  • ********
  • Posts: 7111
    • http://pixinsight.com/
Unable to copy and resample image
« Reply #1 on: 2007 September 17 03:03:26 »
[Footnote: Bueno bueno, se va uno un fin de semana y vaya la que montáis; no se os puede dejar solos  :lol:]

Quote
...the beginProcess/endProcess lines because I have little idea of what that methods do.


A JavaScript script is somewhat like an "unexpected guest" in PixInsight. This is because once you launch a script, its behavior is completely asynchronous with respect to the normal processing workflow in the PixInsight platform. In other words, PI has no way to predict what a script can do and when, two important things that are much more easily controlled with processing modules.

For this reason, I included a number of mechanisms in PixInsight's JavaScript runtime that force a script to behave synchronously when it wants to change an image. If your script needs to modify an image in an existing view, it must inform the core application first. This way, undo/redo buffers can be prepared as necessary, views can be locked to protect pixel data from other threads that can access them concurrently, expensive error recovery and marshaling mechanisms can be triggered, and so on: This is the beginProcess method of View:

Code: [Select]
void View.beginProcess( [undoFlags] )

If you don't call beginProcess(), your script has read-only access to the view. Note that read-only here refers to any property of the view's image, not just to its pixel contents.

The optional undoFlags argument can be specified to control how the core application will prepare to undo things. If you disable generation of a swap file, e.g.:

Code: [Select]
fooBarView.beginProcess( UndoFlag_NoSwapFile );

your script is expected to not modify pixel data in any way; if you do so, please have a good conversation with your users about your reasons.

Having explained beginProcess(), endProcess() is straightforward. Plainly put, if you don't call it anywhere after beginProcess(), PI will bite your ass. :lol:

Quote
This code creates a new ImageWindow and tries to copy the data from another image. As you can see, I've tried several ways Wink to no avail. I've also tried commenting out the beginProcess/endProcess lines because I have little idea of what that methods do. Didn't found a method called .copy, .duplicate or so.


Ok, let's start fixing a problem that you have already reported as a bug. Please copy the following into a new jsh file:

Code: [Select]
#ifndef __PJSR_ResizeMode_jsh
#define __PJSR_ResizeMode_jsh

/*
 * Resizing modes for Image.resample()
 */
#define ResizeMode_RelativeDimensions  0  // Resize relative to current image dimensions
#define ResizeMode_AbsolutePixels      1  // Resize to absolute dimensions in pixels
#define ResizeMode_AbsoluteCentimeters 2  // Resize to absolute dimensions in centimeters
#define ResizeMode_AbsoluteInches      3  // Resize to absolute dimensions in inches
#define ResizeMode_ForceArea           4  // Force the total number of pixels and keep existing aspect ratio
#define ResizeMode_Default             ResizeMode_RelativeDimensions

/*
 * Absolute resizing modes for Image.resample()
 *
 * These modes are only applicable when the main resize mode is
 * ResizeMode_AbsolutePixels, ResizeMode_AbsoluteCentimeters or
 * ResizeMode_AbsoluteInches.
 */
#define AbsoluteResizeMode_ForceWidthAndHeight  0  // Force both dimensions
#define AbsoluteResizeMode_ForceWidth           1  // Force width, preserve aspect ratio
#define AbsoluteResizeMode_ForceHeight          2  // Force height, preserve aspect ratio
#define AbsoluteResizeMode_Default = AbsoluteResizeMode_ForceWidthAndHeight

#endif   // __PJSR_ResizeMode_jsh


and save it as <pjsr/ResizeMode.jsh>. This file is missing in the current PJSR distribution, and the corresponding constants haven't been included in the object browser interface. This of course will be fixed in the next version.

Now here's the code that will do what you're trying to achieve:

Code: [Select]
#include <pjsr/Interpolation.jsh>
#include <pjsr/ResizeMode.jsh>
#include <pjsr/SampleType.jsh>
#include <pjsr/UndoFlag.jsh>

function test()
{
   var w = ImageWindow.activeWindow;
   if ( w.isNull )
      throw Error( "No image window." );

   var v = w.mainView;

   var copy_img = new ImageWindow (
      1, 1, 1, v.image.bitsPerSample,
      v.image.sampleType == SampleType_Real,
      false, "foobar23204" );

   with ( copy_img.mainView )
   {
       beginProcess( UndoFlag_NoSwapFile );
       image.assign( v.image );
       image.resample( 640, 480, Interpolation_BicubicBSpline, ResizeMode_AbsolutePixels );
       //image.resample( 640, 1, Interpolation_BicubicBSpline, ResizeMode_AbsolutePixels, AbsoluteResizeMode_ForceWidth );
       endProcess();
   }

   copy_img.show();
   copy_img.zoomToFit();
}

test();


The new image window (copy_img) doesn't need to be of the same dimensions and color space as the source image that you want to duplicate. Always create a 1x1x1 image window, then assign the source image to its main view into a beginProcess()...endProcess() sequence. A 1x1x1 image used in this way will reduce your script's memory requirements.

Note that you can resize an image mainly in two ways: relative to its current dimensions and absolute. Relative is the default mode.

When resizing relatively, the first two arguments of Image.resize() are (positive) real scaling ratios that will multiply current dimensions. When resizing in absolute mode, you can force width, height or both in several ways. I think that the ResizeMode.jsh explains this well.

A potential pitfall: be aware that something like:

Code: [Select]
foo.resize( 640, 480, Interpolation_Bicubic );

will likely be an error since you're asking PI to multiply current horizontal and vertical dimensions by 640 and 480, respectively. If you apply the above code to a moderately sized image, say 1024x1024, go imagine... some 3.7x10^12 bytes for a 32-bit RGB color image! I'm not sure of how memory management routines would handle such a request because it goes well beyond 32-bit numbers... so better don't write things like that.
Juan Conejero
PixInsight Development Team
http://pixinsight.com/

Offline David Serrano

  • PTeam Member
  • PixInsight Guru
  • ****
  • Posts: 503
Unable to copy and resample image
« Reply #2 on: 2007 September 17 04:21:11 »
Quote from: "Juan Conejero"
[...], views can be locked to protect pixel data from other threads that can access them concurrently, [...]


Hmmm I actually suspected this one ;). The joy of synchronism...

Nice explanation. The UndoFlags then could be considered as a declaration of intentions: "I plan to modify the FITS keywords and the identifier of this image, so let's beginProcess (UndoFlag_Keywords | UndoFlag_ImageId)". Right?


Quote from: "Juan Conejero"
Always create a 1x1x1 image window, then assign the source image to its main view into a beginProcess()...endProcess() sequence.


Yeah, I started like that but gradually gave in while trying to make it work. Thanks for the note, it's nice to know.


Quote from: "Juan Conejero"
Relative is the default mode.


That doesn't surprise me, since it can be seen in the Resample interface. I should've realized that I was in fact scaling the image and overflowing all available memory. But, since neither valgrind nor strace gave me any clues about it being a memory problem, I didn't.

So I was actually doing things well (I was going to use AbsoluteResizeMode_ForceWidth, like you showed), which is comforting :).
--
 David Serrano

Offline Juan Conejero

  • PTeam Member
  • PixInsight Jedi Grand Master
  • ********
  • Posts: 7111
    • http://pixinsight.com/
Unable to copy and resample image
« Reply #3 on: 2007 September 17 04:39:16 »
Quote
The UndoFlags then could be considered as a declaration of intentions: "I plan to modify the FITS keywords and the identifier of this image, so let's beginProcess (UndoFlag_Keywords | UndoFlag_ImageId)". Right?


Absolutely right. However, note that after endProcess(), PI won't check if your code actually did what it declared to. But of course, don't expect PI to be able to undo what you did if you didn't tell her the truth.

For this reason, we can beginProcess( UndoFlag_NoSwapFile ), then modify pixels in the view's image with impunity. Obviously, we do that only for newly created images; otherwise our users would be less-than-happy ones :->.
Juan Conejero
PixInsight Development Team
http://pixinsight.com/