Subversion Repositories filter_foundry

Rev

Rev 485 | Rev 492 | 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-2009 Toby Thain, toby@telegraphics.com.au
  4.     Copyright (C) 2018-2021 Daniel Marschall, ViaThinkSoft
  5.  
  6.     This program is free software; you can redistribute it and/or modify
  7.     it under the terms of the GNU General Public License as published by
  8.     the Free Software Foundation; either version 2 of the License, or
  9.     (at your option) any later version.
  10.  
  11.     This program is distributed in the hope that it will be useful,
  12.     but WITHOUT ANY WARRANTY; without even the implied warranty of
  13.     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14.     GNU General Public License for more details.
  15.  
  16.     You should have received a copy of the GNU General Public License
  17.     along with this program; if not, write to the Free Software
  18.     Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
  19. */
  20.  
  21. /* Win32 user interface routines */
  22.  
  23. #include "world.h"
  24.  
  25. #include "PIAbout.h"
  26.  
  27. #include <windows.h>
  28. #include <commctrl.h>
  29.  
  30. #include "ff.h"
  31. #include "slider_win.h"
  32. #include "version.h"
  33.  
  34. HWND preview_hwnd;
  35. HCURSOR hCurHandOpen;
  36. HCURSOR hCurHandGrab;
  37.  
  38. HCURSOR hCurHandQuestion;
  39. HICON hIconCautionSign;
  40.  
  41. extern HINSTANCE hDllInstance;
  42.  
  43. void DoAbout(AboutRecordPtr pb){
  44.         char text[1000];
  45.         char title[200];
  46.         PlatformData *p = (PlatformData*)pb->platformData;
  47.  
  48.         if (gdata && gdata->standalone) {
  49.                 sprintf(title, "About %s", gdata->parm.szTitle);
  50.                 sprintf(text,  "%s by %s\n" /* {Title} by {Author} */
  51.                                "%s\n" /* {Copyright} */
  52.                                "\n"
  53.                                "This plugin was built using Filter Foundry " VERSION_STR
  54.                                 #ifdef _WIN64
  55.                                " (64 bit)\n"
  56.                                 #else
  57.                                " (32 bit)\n"
  58.                                 #endif
  59.                                "(C) 2003-2009 Toby Thain, 2018-" RELEASE_YEAR " Daniel Marschall\n"
  60.                                "available from " PROJECT_URL,
  61.                                gdata->parm.szTitle,
  62.                                gdata->parm.szAuthor,
  63.                                gdata->parm.szCopyright);
  64.         } else {
  65.                 sprintf(title, "About Filter Foundry");
  66.                 sprintf(text,  "Filter Foundry " VERSION_STR
  67.                                 #ifdef _WIN64
  68.                                " (64 bit)\n"
  69.                                 #else
  70.                                " (32 bit)\n"
  71.                                 #endif
  72.                                "(C) 2003-2009 Toby Thain, 2018-" RELEASE_YEAR " Daniel Marschall\n"
  73.                                "\n"
  74.                                "Latest version available from\n"
  75.                                PROJECT_URL "\n"
  76.                                "\nPlease contact the author with any bug reports,\n"
  77.                                "suggestions or comments.\n"
  78.                                "If you use this program and like it, please consider\n"
  79.                                "making a donation.");
  80.         }
  81.  
  82.         MessageBoxA((HWND)p->hwnd, text, title, MB_TASKMODAL|MB_ICONINFORMATION|MB_OK);
  83. }
  84.  
  85. Boolean simplealert(TCHAR* s){
  86.         HWND hwnd;
  87.         TCHAR title[256] = { 0 };
  88.         if (gdata && gdata->standalone) {
  89.                 #ifdef UNICODE
  90.                 mbstowcs(&title[0], (const char*)gdata->parm.szTitle, 256);
  91.                 #else
  92.                 strcpy(&title[0], gdata->parm.szTitle);
  93.                 #endif
  94.         }
  95.         else {
  96.                 xstrcpy(&title[0], (TCHAR*)TEXT("Filter Foundry"));
  97.         }
  98.         hwnd = gdata ? gdata->hWndMainDlg : NULL;
  99.         return MessageBox(hwnd, s, title, MB_TASKMODAL|MB_ICONERROR|MB_OK) == IDOK;
  100. }
  101.  
  102. Boolean simplewarning(TCHAR* s) {
  103.         HWND hwnd;
  104.         TCHAR title[256] = { 0 };
  105.         if (gdata && gdata->standalone) {
  106.                 #ifdef UNICODE
  107.                 mbstowcs(&title[0], (const char*)gdata->parm.szTitle, 256);
  108.                 #else
  109.                 strcpy(&title[0], gdata->parm.szTitle);
  110.                 #endif
  111.         } else {
  112.                 xstrcpy(&title[0], (TCHAR*)TEXT("Filter Foundry"));
  113.         }
  114.         hwnd = gdata ? gdata->hWndMainDlg : NULL;
  115.         return MessageBox(hwnd,s,title,MB_TASKMODAL|MB_ICONEXCLAMATION|MB_OK) == IDOK;
  116. }
  117.  
  118. Boolean showmessage(TCHAR *s) {
  119.         HWND hwnd;
  120.         TCHAR title[256] = { 0 };
  121.         if (gdata && gdata->standalone) {
  122.                 #ifdef UNICODE
  123.                 mbstowcs(&title[0], (const char*)gdata->parm.szTitle, 256);
  124.                 #else
  125.                 strcpy(&title[0], gdata->parm.szTitle);
  126.                 #endif
  127.         }
  128.         else {
  129.                 xstrcpy(&title[0], (TCHAR*)TEXT("Filter Foundry"));
  130.         }
  131.         hwnd = gdata ? gdata->hWndMainDlg : NULL;
  132.         return MessageBox(hwnd,s,title,MB_TASKMODAL|MB_ICONINFORMATION|MB_OK) == IDOK;
  133. }
  134.  
  135. INT_PTR CALLBACK maindlgproc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam);
  136.  
  137. // Description:
  138. //   Creates a tooltip for an item in a dialog box.
  139. // Parameters:
  140. //   idTool - identifier of an dialog box item.
  141. //   nDlg - window handle of the dialog box.
  142. //   pszText - string to use as the tooltip text.
  143. // Returns:
  144. //   The handle to the tooltip.
  145. //
  146. HWND CreateToolTip(int toolID, HWND hDlg, LPTSTR pszText) {
  147.         // Source: https://docs.microsoft.com/en-us/windows/win32/controls/create-a-tooltip-for-a-control (modified)
  148.  
  149.         HWND hwndTool, hwndTip;
  150.         TOOLINFO toolInfo;
  151.  
  152.         if (!toolID || !hDlg || !pszText) {
  153.                 return FALSE;
  154.         }
  155.         // Get the window of the tool.
  156.         hwndTool = GetDlgItem(hDlg, toolID);
  157.  
  158.         // Create the tooltip. g_hInst is the global instance handle.
  159.         hwndTip = CreateWindowEx(0, TOOLTIPS_CLASS, NULL,
  160.                 WS_POPUP | TTS_ALWAYSTIP /* | TTS_BALLOON*/,
  161.                 CW_USEDEFAULT, CW_USEDEFAULT,
  162.                 CW_USEDEFAULT, CW_USEDEFAULT,
  163.                 hDlg, NULL,
  164.                 hDllInstance, NULL);
  165.  
  166.         if (!hwndTool || !hwndTip) {
  167.                 return (HWND)NULL;
  168.         }
  169.  
  170.         // Associate the tooltip with the tool.
  171.         memset(&toolInfo, 0, sizeof(TOOLINFO)); // toolInfo = { 0 };
  172.         toolInfo.cbSize = sizeof(toolInfo);
  173.         toolInfo.hwnd = hDlg;
  174.         toolInfo.uFlags = TTF_IDISHWND | TTF_SUBCLASS;
  175.         toolInfo.uId = (UINT_PTR)hwndTool;
  176.         toolInfo.lpszText = pszText;
  177.         SendMessage(hwndTip, TTM_ADDTOOL, 0, (LPARAM)&toolInfo);
  178.  
  179.         return hwndTip;
  180. }
  181.  
  182. #define IDT_TIMER_INITPREVIEW_DRAW 1111
  183.  
  184. WNDPROC lpControlEditWndProc[8];
  185.  
  186. LRESULT CALLBACK ControlTextWndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam) {
  187.         extern Boolean doupdates;
  188.         int sliderNum;
  189.         for (sliderNum = 0; sliderNum < 8; sliderNum++) {
  190.                 if (hWnd == GetDlgItem(gdata->hWndMainDlg, FIRSTCTLTEXTITEM + sliderNum)) {
  191.                         if ((uMsg == WM_KEYDOWN) && (wParam == VK_UP) && doupdates)
  192.                         {
  193.                                 uint8_t sliderVal = slider[sliderNum] < 255 ? slider[sliderNum] + 1 : slider[sliderNum];
  194.                                 slider[sliderNum] = sliderVal;
  195.  
  196.                                 SETCTLTEXTINT(gdata->hWndMainDlg, FIRSTCTLTEXTITEM + sliderNum, sliderVal, false);
  197.                                 REPAINTCTL(gdata->hWndMainDlg, FIRSTCTLTEXTITEM + sliderNum);
  198.  
  199.                                 recalc_preview(gpb, gdata->hWndMainDlg);
  200.  
  201.                                 return 0;
  202.                         }
  203.                         if ((uMsg == WM_KEYDOWN) && (wParam == VK_DOWN) && doupdates)
  204.                         {
  205.                                 uint8_t sliderVal = slider[sliderNum] > 0 ? slider[sliderNum] - 1 : slider[sliderNum];
  206.                                 slider[sliderNum] = sliderVal;
  207.  
  208.                                 SETCTLTEXTINT(gdata->hWndMainDlg, FIRSTCTLTEXTITEM + sliderNum, sliderVal, false);
  209.                                 REPAINTCTL(gdata->hWndMainDlg, FIRSTCTLTEXTITEM + sliderNum);
  210.  
  211.                                 recalc_preview(gpb, gdata->hWndMainDlg);
  212.  
  213.                                 return 0;
  214.                         }
  215.                         return CallWindowProc(lpControlEditWndProc[sliderNum], hWnd, uMsg, wParam, lParam);
  216.                 }
  217.         }
  218.         return 0; // should not happen
  219. }
  220.  
  221. INT_PTR CALLBACK maindlgproc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam){
  222.         static POINT origpos;
  223.         static Point origscroll;
  224.         static Boolean panning = false;
  225.  
  226.         int item,i;
  227.         POINT newpos;
  228.         DRAWITEMSTRUCT *pdi;
  229.         Point newscroll = { 0, 0 };
  230.         HGDIOBJ hfnt;
  231.  
  232.         extern Boolean doupdates;
  233.         extern FFBuffer preview_handle;
  234.  
  235.         if ((gdata->pluginDllSliderInfo.initialized) && (wMsg == gdata->pluginDllSliderInfo.messageId)) {
  236.                 // This is for the PLUGIN.DLL sliders only
  237.                 if (doupdates) {
  238.                         int sliderNum = (int)wParam - FIRSTCTLITEM;
  239.                         int sliderVal = (lParam & 0xFFFF);
  240.                         if (sliderVal < 0) sliderVal = 0;
  241.                         else if (sliderVal > 255) sliderVal = 255;
  242.                         slider[sliderNum] = (uint8_t)sliderVal;
  243.  
  244.                         SETCTLTEXTINT(hDlg, FIRSTCTLTEXTITEM + sliderNum, sliderVal, false);
  245.                         REPAINTCTL(hDlg, FIRSTCTLTEXTITEM + sliderNum);
  246.  
  247.                         recalc_preview(gpb, hDlg);
  248.                 }
  249.                 return true;
  250.         }
  251.  
  252.         switch (wMsg) {
  253.         case WM_TIMER:
  254.                 switch (wParam)
  255.                 {
  256.                 case IDT_TIMER_INITPREVIEW_DRAW:
  257.                         KillTimer(hDlg, IDT_TIMER_INITPREVIEW_DRAW);
  258.                         recalc_preview(gpb, hDlg);
  259.                         return 0;
  260.                 }
  261.                 break;
  262.         case WM_INITDIALOG:
  263.                 gdata->hWndMainDlg = hDlg;
  264.  
  265.                 if(gdata->standalone){
  266.                         SetWindowTextA(hDlg,gdata->parm.szTitle); // window title bar
  267.                 }
  268.                 centre_window(hDlg);
  269.  
  270.                 hfnt = GetStockObject(ANSI_FIXED_FONT);
  271.  
  272.                 hCurHandOpen = LoadCursorA(hDllInstance, "HAND_OPEN");
  273.                 hCurHandGrab = LoadCursorA(hDllInstance, "HAND_GRAB");
  274.                 hCurHandQuestion = LoadCursorA(hDllInstance, "HAND_QUESTION");
  275.  
  276.                 hIconCautionSign = LoadIconA(hDllInstance, "CAUTION_ICO");
  277.  
  278.                 // Note: The whole class "Preview" gets the mouse cursor, not just the single item!
  279.                 preview_hwnd = GetDlgItem(hDlg, PREVIEWITEM);
  280.                 GetClientRect(preview_hwnd, &preview_rect);
  281.                 SetClassLongPtr(preview_hwnd, GCLP_HCURSOR, (LONG_PTR)hCurHandOpen);
  282.  
  283.                 // Note: The whole class "Caution" gets the mouse cursor, not just the single item!
  284.                 SetClassLongPtr(GetDlgItem(hDlg, FIRSTICONITEM), GCLP_HCURSOR, (LONG_PTR)hCurHandQuestion);
  285.  
  286.                 for(i = 0; i < 4; ++i){
  287.                         CreateToolTip(FIRSTICONITEM + i, hDlg, (TCHAR*)TEXT("Error in expression! Click to see details."));
  288.                 }
  289.  
  290.                 CreateToolTip(ZOOMINITEM, hDlg, (TCHAR*)TEXT("Zoom in"));
  291.                 CreateToolTip(ZOOMOUTITEM, hDlg, (TCHAR*)TEXT("Zoom out"));
  292.                 CreateToolTip(ZOOMLEVELITEM, hDlg, (TCHAR*)TEXT("Fully zoom in/out"));
  293.  
  294.                 for(i = 0; i < 8; ++i){
  295.                         FF_SetSliderRange(hDlg, FIRSTCTLITEM+i, 0, 255);
  296.                         SendDlgItemMessage(hDlg,FIRSTCTLTEXTITEM+i,     EM_SETLIMITTEXT,3,0);
  297.                 }
  298.                 for(i = 0; i < 4; ++i){
  299.                         SendDlgItemMessage(hDlg,FIRSTEXPRITEM+i,        EM_SETLIMITTEXT,MAXEXPR-1,0); // we need 1 byte as NUL terminator, so our formula can be max 1023
  300.                         SendDlgItemMessage(hDlg,FIRSTEXPRITEM+i,        WM_SETFONT,(WPARAM)hfnt,false);
  301.                 }
  302.  
  303.                 // Implement "up" and "down" keys for the edit controls
  304.                 // TODO: Better use a spin-edit?
  305.                 for (i = 0; i < 8; ++i) {
  306.                         #ifdef _WIN64
  307.                         lpControlEditWndProc[i] = (WNDPROC)SetWindowLongPtr(GetDlgItem(hDlg, FIRSTCTLTEXTITEM + i), GWLP_WNDPROC, (LONG_PTR)&ControlTextWndProc);
  308.                         #else
  309.                         lpControlEditWndProc[i] = (WNDPROC)SetWindowLongPtr(GetDlgItem(hDlg, FIRSTCTLTEXTITEM + i), GWL_WNDPROC, (LONG_PTR)&ControlTextWndProc);
  310.                         #endif
  311.                 }
  312.  
  313.                 maindlginit(hDlg);
  314.  
  315.                 // Win32s (Windows 3.11) compatibility
  316.                 Win32sFixSuperclassing(hDlg, PREVIEWITEM, FIRSTCTLLABELITEM);
  317.                 for (i=0; i<4; i++) Win32sFixSuperclassing(hDlg, FIRSTICONITEM + i, OPENITEM);
  318.  
  319.                 // Some versions of Windows (NT 3.x) won't show the preview if it is calculated here.
  320.                 // So we need to put it in a timer.
  321.                 // Note that 1 millisecond is enough, even if the window needs longer than 1 millisecond to load.
  322.                 //recalc_preview(gpb, hDlg);
  323.                 SetTimer(hDlg, IDT_TIMER_INITPREVIEW_DRAW, 1, (TIMERPROC)NULL);
  324.  
  325.                 break;
  326.         case WM_DESTROY:
  327.                 gdata->hWndMainDlg = 0;
  328.                 DestroyCursor(hCurHandOpen);
  329.                 DestroyCursor(hCurHandGrab);
  330.                 DestroyCursor(hCurHandQuestion);
  331.                 DestroyIcon(hIconCautionSign);
  332.                 break;
  333.         case WM_DRAWITEM:
  334.                 pdi = (DRAWITEMSTRUCT*)lParam;
  335.                 if(pdi->itemAction == ODA_DRAWENTIRE){
  336.                         switch(pdi->CtlID){
  337.                         case PREVIEWITEM:
  338.                                 drawpreview(hDlg,pdi->hDC,lockBuffer(&preview_handle));
  339.                                 unlockBuffer(&preview_handle);
  340.                                 break;
  341.                         case FIRSTICONITEM:
  342.                         case FIRSTICONITEM + 1:
  343.                         case FIRSTICONITEM + 2:
  344.                         case FIRSTICONITEM + 3:
  345.                                 DrawIcon(pdi->hDC, 0, 0, hIconCautionSign);
  346.                                 break;
  347.                         default:
  348.                                 return false;
  349.                         }
  350.                 }else
  351.                         return false; // we couldn't handle the message
  352.                 break;
  353.         case WM_COMMAND:
  354.                 item = LOWORD(wParam);
  355.                 switch(HIWORD(wParam)){
  356.                 //case BN_CLICKED:
  357.                 case STN_CLICKED:
  358.                         // BN_CLICKED = Button clicked
  359.                         // STN_CLICKED = Static controls with SS_NOTIFY clicked
  360.                         // Both have the same ordinal number
  361.                         if(item==PREVIEWITEM && GetCursorPos(&origpos)){
  362.                                 panning = true;
  363.                                 origscroll = preview_scroll;
  364.                                 SetCursor(hCurHandGrab);
  365.                                 SetCapture(hDlg);
  366.                                 break;
  367.                         }
  368.                 /* ... falls through ... */
  369.                 case EN_CHANGE:
  370.                         if(doupdates && !maindlgitem(hDlg,item))
  371.                                 EndDialog(hDlg,item);
  372.                 }
  373.                 break;
  374. //      case WM_LBUTTONDOWN: break;
  375.         case WM_MOUSEMOVE:
  376.                 if(panning && GetCursorPos(&newpos)){
  377.                         newscroll.h = (int16)(origscroll.h - zoomfactor*((double)newpos.x - (double)origpos.x));
  378.                         newscroll.v = (int16)(origscroll.v - zoomfactor*((double)newpos.y - (double)origpos.y));
  379.                         if( newscroll.h != preview_scroll.h || newscroll.v != preview_scroll.v ){
  380.                                 preview_scroll = newscroll;
  381.                                 recalc_preview(gpb,hDlg);
  382.                         }
  383.                 }
  384.                 break;
  385.         case WM_LBUTTONUP:
  386.                 ReleaseCapture();
  387.                 panning = false;
  388.                 break;
  389.         case WM_HSCROLL:
  390.                 // Only for non-Plugin.dll-sliders
  391.                 item = GetDlgCtrlID((HWND)lParam);
  392.                 if(doupdates && !gdata->pluginDllSliderInfo.initialized && item>=FIRSTCTLITEM && item<=FIRSTCTLITEM+7)
  393.                         slidermoved(hDlg,item);
  394.                 break;
  395.         default:
  396.                 return false;
  397.         }
  398.  
  399.         return true;
  400. }
  401.  
  402. Boolean maindialog(FilterRecordPtr pb){
  403.         PlatformData *p;
  404.         INT_PTR res;
  405.  
  406.         // First try to use the sliders from PLUGIN.DLL (only Photoshop)
  407.         if (!Slider_Init_PluginDll(TEXT("FoundrySlider"))) {
  408.                 // If we couldn't get the sliders from PLUGIN.DLL (probably not running in Photoshop),
  409.                 // then try the Microsoft Trackbar Control instead
  410.                 if (!Slider_Init_MsTrackbar(TEXT("FoundrySlider"))) {
  411.                         // This will happen if we neither have PLUGIN.DLL, nor the Microsoft Trackbar Control (msctls_trackbar32).
  412.                         // "msctls_trackbar32" is not included in Windows NT 3.1, and since there is no OCX or RegSvr32.
  413.                         // It is included in Windows NT 3.5x.
  414.  
  415.                         //simplealert((TCHAR*)TEXT("This plugin requires Photoshop's PLUGIN.DLL or the Microsoft Trackbar Control (msctls_trackbar32) which was not found on your system."));
  416.                         //return false;
  417.  
  418.                         // We simply hide the sliders and let the user enter the numeric values in the edit-box.
  419.                         simplewarning((TCHAR*)TEXT("Visual sliders are not available because neither PLUGIN.DLL, nor the Microsoft Trackbar Control (msctls_trackbar32) was found on your system."));
  420.                         Slider_Init_None(TEXT("FoundrySlider"));
  421.                 }
  422.         }
  423.  
  424.         // For the preview image and caution symbols, we register subclasses, so that we can assign a mouse cursor to this class.
  425.         MakeSimpleSubclass(TEXT("Preview"), WC_STATIC);
  426.         MakeSimpleSubclass(TEXT("Caution"), WC_BUTTON);
  427.  
  428.         // Now show the dialog
  429.         p = (PlatformData*)pb->platformData;
  430.  
  431.         // Note: "Invalid Cursor Handle" is the error when an unrecognized control class is detected
  432.         res = DialogBoxParam(hDllInstance,MAKEINTRESOURCE(gdata->standalone ? ID_PARAMDLG : ID_MAINDLG),
  433.                              (HWND)p->hwnd,maindlgproc,0);
  434.         if (res == 0) {
  435.                 simplealert((TCHAR*)TEXT("DialogBoxParam invalid parent window handle"));
  436.         }
  437.         if (res == -1) {
  438.                 TCHAR s[0x300];
  439.                 xstrcpy(s, (TCHAR*)TEXT("DialogBoxParam failed: "));
  440.                 FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError(), 0, s + xstrlen(s), 0x300 - (DWORD)xstrlen(s), NULL);
  441.                 simplealert(&s[0]);
  442.         }
  443.  
  444.         // Clean up after the dialog has been closed
  445.         UnregisterClass(TEXT("Preview"), hDllInstance);
  446.         UnregisterClass(TEXT("Caution"), hDllInstance);
  447.         UnregisterClass(TEXT("FoundrySlider"), hDllInstance);
  448.         Slider_Uninit_PluginDll();
  449.         Slider_Uninit_MsTrackbar();
  450.         Slider_Uninit_None();
  451.  
  452.         return res == IDOK;
  453. }
  454.