Subversion Repositories filter_foundry

Rev

Rev 110 | Rev 119 | 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-7 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 "preview.h"
  23.  
  24. #ifdef MAC_ENV
  25.         #include <fp.h>
  26. #endif
  27. #include <math.h>
  28.  
  29. #include "PIProperties.h"
  30.  
  31. extern FilterRecordPtr gpb;
  32.  
  33. PSPixelMap preview_pmap;
  34. PSPixelMask preview_pmask;
  35. Handle preview_handle;
  36. UIRECT preview_rect;
  37. int preview_w,preview_h,previewerr = false,needall = false,needinput = true;
  38. Point preview_scroll;
  39. Boolean preview_complete = false;
  40. double zoomfactor,fitzoom;
  41.  
  42. Boolean setup_preview(FilterRecordPtr pb, int nplanes){
  43.         double zh,zv;
  44.  
  45.         if(pb->displayPixels && pb->advanceState){
  46.                 preview_w = MIN(preview_rect.right - preview_rect.left - 2,
  47.                                                 pb->filterRect.right - pb->filterRect.left);
  48.                 preview_h = MIN(preview_rect.bottom - preview_rect.top - 2,
  49.                                                 pb->filterRect.bottom - pb->filterRect.top);
  50.                 zh = (pb->filterRect.right - pb->filterRect.left)/(double)preview_w;
  51.                 zv = (pb->filterRect.bottom - pb->filterRect.top)/(double)preview_h;
  52.                 fitzoom = zh > zv ? zh : zv;
  53.  
  54.                 preview_pmap.version = 1;
  55.                 preview_pmap.bounds.left = preview_pmap.bounds.top = 0;
  56.                 preview_pmap.bounds.right = preview_w;
  57.                 preview_pmap.bounds.bottom = preview_h;
  58.                 preview_pmap.imageMode = pb->imageMode;
  59.                 preview_pmap.rowBytes = nplanes*preview_w;
  60.                 preview_pmap.colBytes = nplanes;
  61.                 preview_pmap.planeBytes = 1; /*interleaved*/
  62.         //      preview_pmap.baseAddr = preview_data;
  63.         /* baseAddr must be set before using pixelmap */
  64.  
  65.                 //---------------------------------------------------------------------------
  66.                 // Fields new in version 1:
  67.                 //---------------------------------------------------------------------------
  68.                 preview_pmap.mat = NULL;
  69.  
  70.                 if( (pb->imageMode == plugInModeRGBColor && nplanes == 4)
  71.                  || (pb->imageMode == plugInModeLabColor && nplanes == 4)
  72.                  || (pb->imageMode == plugInModeGrayScale && nplanes == 2)
  73.                  || (pb->imageMode == plugInModeDuotone && nplanes == 2) )
  74.                 {
  75.                         preview_pmask.next = NULL;
  76.         //              preview_pmask.maskData = preview_data+3;
  77.                         preview_pmask.rowBytes = preview_pmap.rowBytes;
  78.                         preview_pmask.colBytes = nplanes;
  79.                         preview_pmask.maskDescription = kSimplePSMask;
  80.                         preview_pmap.masks = &preview_pmask;
  81.                 }else
  82.                         preview_pmap.masks = NULL;
  83.  
  84.                 preview_handle = PINEWHANDLE((long)preview_h * preview_pmap.rowBytes);
  85.         }else
  86.                 preview_handle = NULL;
  87.         return preview_handle != NULL;
  88.  
  89.         //---------------------------------------------------------------------------
  90.         // Fields new in version 2:
  91.         //---------------------------------------------------------------------------
  92. //      preview_pmap.pixelOverlays;
  93. //      preview_pmap.colorManagementOptions;
  94. }
  95.  
  96. void dispose_preview(){
  97.         if(preview_handle){
  98.                 PIDISPOSEHANDLE(preview_handle);
  99.                 preview_handle = NULL;
  100.         }
  101. }
  102.  
  103. void recalc_preview(FilterRecordPtr pb,DIALOGREF dp){
  104.         OSErr e;
  105.         int j,n,scaledw,scaledh,imgw,imgh;
  106.         Rect r,outRect;
  107.         Ptr outrow;
  108.         double cor = 0.00;
  109.  
  110.         preview_complete = false;
  111.  
  112.         if(preview_handle){
  113.                 /* size of previewed area, of source image; but no larger than filtered area (selection) */
  114.                 scaledw = zoomfactor*preview_w;
  115.                 if(scaledw > (pb->filterRect.right - pb->filterRect.left))
  116.                         scaledw = pb->filterRect.right - pb->filterRect.left;
  117.                 scaledh = zoomfactor*preview_h;
  118.                 if(scaledh > (pb->filterRect.bottom - pb->filterRect.top))
  119.                         scaledh = pb->filterRect.bottom - pb->filterRect.top;
  120.  
  121.                 /* scale clipped preview area down again - this becomes the pixel size of preview */
  122.                 imgw = scaledw/zoomfactor;
  123.                 if(imgw > preview_w)
  124.                         imgw = preview_w;
  125.                 imgh = scaledh/zoomfactor;
  126.                 if(imgh > preview_h)
  127.                         imgh = preview_h;
  128.  
  129.                 /* compute source data rectangle (inRect) */
  130.  
  131.                 /* centre preview on filtered part of input image, adding scroll offset */
  132.                 r.left = (pb->filterRect.left + pb->filterRect.right - scaledw)/2 + preview_scroll.h;
  133.                 /* make sure it does not go outside the input area */
  134.                 if(r.left < pb->filterRect.left) {
  135.                         cor = pb->filterRect.left - r.left;
  136.                         r.left = pb->filterRect.left;
  137.                 }
  138.                 else if(r.left+scaledw > pb->filterRect.right) {
  139.                         cor = pb->filterRect.right - (r.left+scaledw);
  140.                         r.left = pb->filterRect.right - scaledw;
  141.                 } else {
  142.                         cor = 0;
  143.                 }
  144.                 r.right = r.left + scaledw;
  145.                 preview_pmap.maskPhaseCol = (preview_scroll.h+cor)/zoomfactor; // phase of the checkerboard
  146.  
  147.                 /* now compute for vertical */
  148.                 r.top = (pb->filterRect.top + pb->filterRect.bottom - scaledh)/2 + preview_scroll.v;
  149.                 if(r.top < pb->filterRect.top) {
  150.                         cor = pb->filterRect.top - r.top;
  151.                         r.top = pb->filterRect.top;
  152.                 }
  153.                 else if(r.top+scaledh > pb->filterRect.bottom) {
  154.                         cor = pb->filterRect.bottom - (r.top+scaledh);
  155.                         r.top = pb->filterRect.bottom - scaledh;
  156.                 } else {
  157.                         cor = 0;
  158.                 }
  159.                 r.bottom = r.top + scaledh;
  160.                 preview_pmap.maskPhaseRow = (preview_scroll.v+cor)/zoomfactor; // phase of the checkerboard
  161.  
  162.                 /* if formulae need random access to image - src(), rad() - we must request entire area: */
  163.                 if(needall)
  164.                         SETRECT(pb->inRect,0,0,pb->imageSize.h,pb->imageSize.v);
  165.                 else
  166.                         pb->inRect = r;
  167.  
  168.                 pb->outRect = pb->inRect;
  169.                 SETRECT(pb->maskRect,0,0,0,0);
  170.                 pb->inLoPlane = pb->outLoPlane = 0;
  171.                 pb->inHiPlane = pb->outHiPlane = nplanes-1;
  172.  
  173.                 if( !needinput || !(e = pb->advanceState()) ){
  174.                         Ptr outptr = PILOCKHANDLE(preview_handle,false);
  175.                         int blankrows = (preview_h - imgh)/2,
  176.                                 blankcols = (preview_w - imgw)/2,
  177.                                 pmrb = preview_pmap.rowBytes;
  178.  
  179.                         evalinit();
  180.  
  181.                         SETRECT(outRect,0,0,imgw,imgh);
  182.  
  183.                         e = process_scaled(pb, false, &r, &outRect,
  184.                                         outptr + pmrb*blankrows + nplanes*blankcols, pmrb, zoomfactor);
  185.                         if(blankrows){
  186.                                 memset(outptr, 0xff, pmrb*blankrows);
  187.                                 n = preview_h - blankrows - imgh; /* blank rows below preview */
  188.                                 memset(outptr + pmrb*(blankrows+imgh), 0xff, pmrb*n);
  189.                         }
  190.                         if(blankcols){
  191.                                 n = preview_w - blankcols - imgw; /* blank columns on right side of preview */
  192.                                 outrow = outptr + pmrb*blankrows;
  193.                                 for(j = blankrows; j < preview_h - blankrows; ++j){
  194.                                         memset(outrow, 0xff, nplanes*blankcols);
  195.                                         memset(outrow + nplanes*(blankcols+imgw), 0xff, nplanes*n);
  196.                                         outrow += pmrb;
  197.                                 }
  198.                         }
  199.  
  200.                         if(!e){
  201.                                 preview_complete = true;
  202.  
  203. #ifdef WIN_ENV
  204.                                 {
  205.                                 extern HWND preview_hwnd;
  206.                                 HDC hdc = GetDC(preview_hwnd);
  207.  
  208.                                 drawpreview(dp,hdc,outptr);
  209.  
  210.                                 ReleaseDC(preview_hwnd,hdc);
  211.                                 }
  212. #else
  213.                                 {
  214.                                 GrafPtr saveport;
  215.  
  216.                                 GetPort(&saveport);
  217.                                 SetPortDialogPort(dp);
  218.  
  219.                                 drawpreview(dp,NULL,outptr);
  220.  
  221.                                 SetPort(saveport);
  222.                                 }
  223. #endif
  224.                         }
  225.  
  226.                         PIUNLOCKHANDLE(preview_handle);
  227.                 }
  228.  
  229.                 if(e && !previewerr){
  230.                         alertuser("Could not build preview at chosen zoom level.",
  231.                                 e == memFullErr && !needall ? "The image is too large for available memory. Try zooming in.\nIf that does not help, Cancel and retry the filter." : "");
  232.                         previewerr = true;
  233.                 }
  234.  
  235.         }
  236. }
  237.  
  238. OSErr drawpreview(DIALOGREF dp,void *hdc,Ptr imageptr){
  239.         intptr_t watchsusp;
  240.         OSErr e = noErr;
  241.         VRect srcRect;
  242.         UIRECT imagebounds;
  243.  
  244.         if(preview_handle && preview_complete){
  245.  
  246.                 srcRect = preview_pmap.bounds;
  247.  
  248.                 imagebounds.left = (preview_rect.left + preview_rect.right - preview_w)/2;
  249.                 imagebounds.top = (preview_rect.top + preview_rect.bottom - preview_h)/2;
  250.                 imagebounds.right = imagebounds.left + preview_w;
  251.                 imagebounds.bottom = imagebounds.top + preview_h;
  252.  
  253.                 preview_pmap.baseAddr = imageptr;
  254.                 preview_pmask.maskData = imageptr+3; // FIXME: is this offset correct for all modes?!
  255.  
  256.                 if(gpb->propertyProcs->getPropertyProc){
  257.                         gpb->propertyProcs->getPropertyProc(kPhotoshopSignature,propWatchSuspension,0,&watchsusp,NULL);
  258.                         gpb->propertyProcs->setPropertyProc(kPhotoshopSignature,propWatchSuspension,0,watchsusp+1,NULL);
  259.                 }
  260.  
  261.                 e = gpb->displayPixels(&preview_pmap,&srcRect,imagebounds.top,imagebounds.left,hdc);
  262.  
  263.                 if(gpb->propertyProcs->getPropertyProc)
  264.                         gpb->propertyProcs->setPropertyProc(kPhotoshopSignature,propWatchSuspension,0,watchsusp,NULL);
  265.         }
  266.         return e;
  267. }
  268.