Subversion Repositories filter_foundry

Rev

Rev 8 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

  1. /*
  2.         This file is part of Filter Foundry, a filter plugin for Adobe Photoshop
  3.     Copyright (C) 2003-5 Toby Thain, toby@telegraphics.com.au
  4.  
  5.     This program is free software; you can redistribute it and/or modify
  6.     it under the terms of the GNU General Public License as published by  
  7.     the Free Software Foundation; either version 2 of the License, or
  8.     (at your option) any later version.
  9.  
  10.     This program is distributed in the hope that it will be useful,
  11.     but WITHOUT ANY WARRANTY; without even the implied warranty of
  12.     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  13.     GNU General Public License for more details.
  14.  
  15.     You should have received a copy of the GNU General Public License  
  16.     along with this program; if not, write to the Free Software
  17.     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  18. */
  19.  
  20. /* This is PLATFORM INDEPENDENT user interface code - mainly dialog logic */
  21.  
  22. #include "ff.h"
  23.  
  24. #include "piproperties.h"
  25.  
  26. #include "node.h"
  27. #include "parser.h"
  28. #include "funcs.h"
  29. #include "y.tab.h"
  30. #include "choosefile.h"
  31.  
  32. PSPixelMap preview_pmap;
  33. PSPixelMask preview_pmask;
  34. Handle preview_handle;
  35. UIRECT preview_rect;
  36. int preview_w,preview_h;
  37. Point preview_scroll;
  38. Boolean preview_complete = false;
  39.  
  40. Boolean setup_preview(FilterRecordPtr pb){
  41.  
  42.         //dbg("setup_preview");
  43.        
  44.         if(pb->displayPixels && pb->advanceState){
  45.  
  46.                 preview_w = MIN(preview_rect.right - preview_rect.left - 2,pb->filterRect.right - pb->filterRect.left);
  47.                 preview_h = MIN(preview_rect.bottom - preview_rect.top - 2,pb->filterRect.bottom - pb->filterRect.top);
  48.        
  49.                 preview_pmap.version = 1;
  50.                 preview_pmap.bounds.left = preview_pmap.bounds.top = 0;
  51.                 preview_pmap.bounds.right = preview_w;
  52.                 preview_pmap.bounds.bottom = preview_h;
  53.                 preview_pmap.imageMode = nplanes>1 ? plugInModeRGBColor : plugInModeGrayScale;
  54.                 preview_pmap.rowBytes = nplanes*preview_w;
  55.                 preview_pmap.colBytes = nplanes;
  56.                 preview_pmap.planeBytes = 1; /*interleaved*/
  57.         //      preview_pmap.baseAddr = preview_data;
  58.         /* baseAddr must be set before using pixelmap */
  59.                
  60.                 //---------------------------------------------------------------------------
  61.                 // Fields new in version 1:
  62.                 //---------------------------------------------------------------------------  
  63.                 preview_pmap.mat = NULL;
  64.                
  65.                 if(nplanes==4){
  66.                         preview_pmask.next = NULL;
  67.         //              preview_pmask.maskData = preview_data+3;
  68.                         preview_pmask.rowBytes = preview_pmap.rowBytes;
  69.                         preview_pmask.colBytes = nplanes;
  70.                         preview_pmask.maskDescription = kSimplePSMask;
  71.                         preview_pmap.masks = &preview_pmask;
  72.                 }else
  73.                         preview_pmap.masks = NULL;
  74.        
  75.                 preview_handle = PINEWHANDLE((long)preview_h * preview_pmap.rowBytes);
  76.         }else
  77.                 preview_handle = NULL;
  78.         return preview_handle != NULL;
  79.  
  80.         //---------------------------------------------------------------------------
  81.         // Fields new in version 2:
  82.         //---------------------------------------------------------------------------  
  83. //      preview_pmap.pixelOverlays;
  84. //      preview_pmap.colorManagementOptions;
  85.  
  86. //      setup(pb); // prepare for evaluations
  87. }
  88.  
  89. void dispose_preview(){
  90.         if(preview_handle){
  91.                 PIDISPOSEHANDLE(preview_handle);
  92.                 preview_handle = NULL;
  93.         }
  94. }
  95.  
  96. void recalc_preview(FilterRecordPtr pb,DIALOGREF dp){
  97.         extern int srcradused,needinput;
  98.         extern double zoomfactor;
  99.         OSErr e;
  100.         int j,n,scaledw,scaledh,imgw,imgh;
  101.         Rect r,outRect;
  102.         Ptr outrow;
  103.  
  104.         if(preview_handle){
  105.  
  106.                 /* size of previewed area, of source image; but no larger than filtered area (selection) */
  107.                 scaledw = zoomfactor*preview_w;
  108.                 if(scaledw > (pb->filterRect.right - pb->filterRect.left))
  109.                         scaledw = (pb->filterRect.right - pb->filterRect.left);
  110.                 scaledh = zoomfactor*preview_h;
  111.                 if(scaledh > (pb->filterRect.bottom - pb->filterRect.top))
  112.                         scaledh = (pb->filterRect.bottom - pb->filterRect.top);
  113.                        
  114.                 /* scale clipped preview area down again - this becomes the pixel size of preview */
  115.                 imgw = scaledw/zoomfactor;
  116.                 if(imgw > preview_w)
  117.                         imgw = preview_w;
  118.                 imgh = scaledh/zoomfactor;
  119.                 if(imgh > preview_h)
  120.                         imgh = preview_h;
  121.  
  122.                 // Use to set the phase of the checkerboard:
  123.                 preview_pmap.maskPhaseRow = preview_scroll.v/zoomfactor;
  124.                 preview_pmap.maskPhaseCol = preview_scroll.h/zoomfactor;
  125.  
  126.                 /* compute source data rectangle (inRect) */
  127.  
  128.                 /* centre preview on filtered part of input image, adding scroll offset */
  129.                 r.left = (pb->filterRect.left+pb->filterRect.right-scaledw)/2 + preview_scroll.h;
  130.                 /* make sure it does not go outside the input area */
  131.                 if(r.left < pb->filterRect.left)
  132.                         r.left = pb->filterRect.left;
  133.                 else if(r.left > pb->filterRect.right-scaledw)
  134.                         r.left = pb->filterRect.right-scaledw;
  135.                 r.right = r.left + scaledw;
  136.  
  137.                 /* now compute for vertical */
  138.                 r.top = (pb->filterRect.top+pb->filterRect.bottom-scaledh)/2 + preview_scroll.v;
  139.                 if(r.top < pb->filterRect.top)
  140.                         r.top = pb->filterRect.top;
  141.                 else if(r.top > pb->filterRect.bottom-scaledh)
  142.                         r.top = pb->filterRect.bottom-scaledh;
  143.                 r.bottom = r.top + scaledh;
  144.  
  145.                 /* if formulae need random access to image - src(), rad() - we must request entire area: */
  146.                 if(srcradused){
  147.                         SETRECT(pb->inRect,0,0,pb->imageSize.h,pb->imageSize.v);
  148.                 }else
  149.                         pb->inRect = r;
  150.                
  151.                 pb->outRect = pb->inRect;
  152.                 SETRECT(pb->maskRect,0,0,0,0);
  153.                 pb->inLoPlane = pb->outLoPlane = 0;
  154.                 pb->inHiPlane = pb->outHiPlane = nplanes-1;
  155.                                
  156. //dbg("recalc_preview: about to call advanceState()");
  157.                 if( !needinput || !(e = pb->advanceState()) ){
  158.                         Ptr outptr = PILOCKHANDLE(preview_handle,false);
  159.                         int blankrows = (preview_h-imgh)/2,
  160.                                 blankcols = (preview_w-imgw)/2;
  161.  
  162.                         INITRANDSEED();
  163. //dbg("recalc_preview: about to call process()");
  164.                                
  165.                         SETRECT(outRect,0,0,imgw,imgh);
  166.                        
  167.                         e = process_scaled(pb,false,
  168.                                         &pb->inRect,&r,&outRect,
  169.                                         outptr+preview_pmap.rowBytes*blankrows+nplanes*blankcols,preview_pmap.rowBytes,
  170.                                         zoomfactor);
  171.                         if(blankrows){
  172.                                 memset(outptr,0xff,preview_pmap.rowBytes*blankrows);
  173.                                 n = preview_h - blankrows - imgh; /* blank rows below preview */
  174.                                 memset(outptr+preview_pmap.rowBytes*(blankrows+imgh),0xff,preview_pmap.rowBytes*n);
  175.                         }
  176.                         if(blankcols){
  177.                                 n = preview_w - blankcols - imgw; /* blank columns on right side of preview */
  178.                                 outrow = outptr+preview_pmap.rowBytes*blankrows;
  179.                                 for( j = blankrows ; j < preview_h - blankrows ; ++j ){
  180.                                         memset(outrow,0xff,nplanes*blankcols);
  181.                                         memset(outrow+nplanes*(blankcols+imgw),0xff,nplanes*n);
  182.                                         outrow += preview_pmap.rowBytes;
  183.                                 }
  184.                         }
  185.  
  186.                         if(!e){
  187.                                 preview_complete = true;
  188.  
  189. #ifdef WIN_ENV
  190.                                 {
  191.                                 extern HWND preview_hwnd;
  192.                                 HDC hdc = GetDC(preview_hwnd);
  193.                                
  194. //dbg("recalc_preview: about to call drawpreview()");
  195.                                 drawpreview(dp,hdc,outptr);
  196.                                
  197.                                 ReleaseDC(preview_hwnd,hdc);
  198.                                 }
  199. #else
  200.                                 {
  201.                                 GrafPtr saveport;
  202.  
  203.                                 GetPort(&saveport);
  204.                                 SetPortDialogPort(dp);
  205.                                
  206.                                 drawpreview(dp,NULL,outptr);
  207.                                
  208.                                 SetPort(saveport);
  209.                                 }
  210. #endif
  211.                         }
  212.  
  213.                         PIUNLOCKHANDLE(preview_handle);
  214.  
  215.                 }else{ char s[0x100];
  216.                         sprintf(s,"recalc_preview: advanceState failed (%d)\n\
  217. inRect=(%d,%d,%d,%d) filterRect=(%d,%d,%d,%d) inLoPlane=%d inHiPlane=%d ",
  218.                                 e,
  219.                                 pb->inRect.left,pb->inRect.top,pb->inRect.right,pb->inRect.bottom,
  220.                                 pb->filterRect.left,pb->filterRect.top,pb->filterRect.right,pb->filterRect.bottom,
  221.                                 pb->inLoPlane,pb->inHiPlane);
  222.                         dbg(s);
  223.                 }
  224.         }
  225. }
  226.  
  227. OSErr drawpreview(DIALOGREF dp,void *hdc,Ptr imageptr){
  228.         int32 watchsusp;
  229.         OSErr e = noErr;
  230.         VRect srcRect;
  231.         UIRECT imagebounds;
  232.  
  233.         if(preview_handle && preview_complete){
  234.  
  235.                 srcRect = preview_pmap.bounds;
  236.  
  237.                 imagebounds.left = (preview_rect.left+preview_rect.right-preview_w)/2;
  238.                 imagebounds.top = (preview_rect.top+preview_rect.bottom-preview_h)/2;
  239.                 imagebounds.right = imagebounds.left + preview_w;
  240.                 imagebounds.bottom = imagebounds.top + preview_h;
  241.  
  242.                 preview_pmap.baseAddr = imageptr;//PILOCKHANDLE(preview_handle,false);
  243.                 preview_pmask.maskData = imageptr+3;
  244.  
  245.                 if(gpb->propertyProcs->getPropertyProc){
  246.                         gpb->propertyProcs->getPropertyProc(kPhotoshopSignature,propWatchSuspension,0,&watchsusp,NULL);
  247.                         gpb->propertyProcs->setPropertyProc(kPhotoshopSignature,propWatchSuspension,0,watchsusp+1,NULL);
  248.                 }
  249.                 e = gpb->displayPixels(&preview_pmap,&srcRect,imagebounds.top,imagebounds.left,hdc);
  250.  
  251.                 if(gpb->propertyProcs->getPropertyProc)
  252.                         gpb->propertyProcs->setPropertyProc(kPhotoshopSignature,propWatchSuspension,0,watchsusp,NULL);
  253.         }
  254.         return e;
  255. }
  256.  
  257. #if 0
  258. // old process-then-scale preview code
  259.                         if(scaleddata = PINEWHANDLE((long)nplanes*scaledw*scaledh)){
  260.                                 Ptr scaledptr = PILOCKHANDLE(scaleddata,false);
  261.                                 //process(pb,false,&pb->inRect,&r,&r,outptr,preview_pmap.rowBytes)
  262.                                 long scaledrb = (long)nplanes*scaledw;
  263.  
  264.                                 if(!(e = process(pb,false,&pb->inRect,&r,&r,scaledptr,scaledrb)){
  265.        
  266.                                         /* copy data from scaledptr to outptr, scaling as we go */
  267.                                         if(zoomfactor == 1.)
  268.                                                 memcpy(outptr,scaledptr,(long)nplanes*preview_w*preview_h);
  269.                                         else{
  270.                                                 int blankrows = (preview_h-imgh)/2,
  271.                                                         blankcols = (preview_w-imgw)/2;
  272.        
  273.                                                 memset(outptr,0xff,preview_pmap.rowBytes*blankrows);
  274.                                                 memset(outptr+preview_pmap.rowBytes*(preview_h-blankrows),0xff,preview_pmap.rowBytes*blankrows);
  275.        
  276.                                                 outrow = outptr+preview_pmap.rowBytes*blankrows;
  277.                                                 for( j=blankrows,y=0. ; j<preview_h-blankrows ; ++j,y+=zoomfactor ){
  278.                                                         memset(outrow,0xff,nplanes*blankcols);
  279.                                                         memset(outrow+nplanes*(preview_w-blankcols),0xff,nplanes*blankcols);
  280.                                                         outrow += nplanes*blankcols;
  281.                                                         inrow = scaledptr + scaledrb*(int)y;
  282.                                                         for( i=blankcols,x = 0. ; i<(preview_w-blankcols) ; ++i,x+=zoomfactor){
  283.                                                                 inpix = inrow + nplanes*(int)x ;
  284.                                                                 for(k=0;k<nplanes;++k)
  285.                                                                         *outrow++ = inpix[k];
  286.                                                         }
  287.                                                         outrow += nplanes*blankcols;
  288.                                                 }
  289.                                         }
  290.                                 }
  291.                                 PIDISPOSEHANDLE(scaleddata);
  292.                         }else e = memFullErr;
  293. #endif
  294.