Subversion Repositories Games.Chess Giants

Rev

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

  1. // dialog_registration.cpp
  2.  
  3. #include "../common.h"
  4.  
  5.  
  6. // dialog template
  7. #define THIS_DIALOG DIALOG_REGISTER
  8.  
  9.  
  10. // global variables used in this module only
  11. static wchar_t candidate_email[128] = L"";
  12. static unsigned __int32 candidate_code = 0; // 32 bits
  13.  
  14.  
  15. // prototypes of local functions
  16. static int CALLBACK DialogProc_ThisDialog (HWND hWnd, unsigned int message, WPARAM wParam, LPARAM lParam);
  17. static unsigned __int32 RetrieveActivationCode (wchar_t *candidate_email, wchar_t **failure_reason);
  18. static int recv_with_timeout (int socket_id, float timeout_in_seconds, char *outbuf, size_t outbuf_size, int flags);
  19.  
  20.  
  21. void DialogBox_Registration (void)
  22. {
  23.    // helper function to fire up the modal dialog box
  24.  
  25.    // display the dialog box (set the parent to be the desktop)
  26.    DialogBox (hAppInstance, MAKEINTRESOURCE (THIS_DIALOG), NULL, DialogProc_ThisDialog);
  27.  
  28.    return; // finished processing this dialog
  29. }
  30.  
  31.  
  32. static int CALLBACK DialogProc_ThisDialog (HWND hWnd, unsigned int message, WPARAM wParam, LPARAM lParam)
  33. {
  34.    // message handler for the dialog box
  35.  
  36.    static wchar_t temp_string[1024];
  37.  
  38.    unsigned short wParam_hiword;
  39.    unsigned short wParam_loword;
  40.    wchar_t *failure_reason;
  41.    int write_index;
  42.    int read_index;
  43.    int length;
  44.  
  45.    // filter out the commonly used message values
  46.    wParam_hiword = HIWORD (wParam);
  47.    wParam_loword = LOWORD (wParam);
  48.  
  49.    // have we just fired up this window ?
  50.    if (message == WM_INITDIALOG)
  51.    {
  52.       // center the window
  53.       CenterWindow (hWnd, hMainWnd);
  54.  
  55.       // set dialog icons (small one for title bar & big one for task manager)
  56.       SendMessage (hWnd, WM_SETICON, ICON_SMALL, (LPARAM) LoadIcon (hAppInstance, MAKEINTRESOURCE (ICON_MAIN)));
  57.       SendMessage (hWnd, WM_SETICON, ICON_BIG, (LPARAM) LoadIcon (hAppInstance, MAKEINTRESOURCE (ICON_MAIN)));
  58.  
  59.       // set window title and control texts
  60.       SetWindowText (hWnd, LOCALIZE (L"Registration_Title"));
  61.  
  62.       // set the static texts
  63.       Static_SetText (GetDlgItem (hWnd, STATICTEXT_REGISTRATION_QUESTION), LOCALIZE (L"Registration_Question"));
  64.       Static_SetText (GetDlgItem (hWnd, STATICTEXT_REGISTRATION_INSTRUCTIONS), LOCALIZE (L"Registration_Instructions"));
  65.       Static_SetText (GetDlgItem (hWnd, STATICTEXT_REGISTRATION_EMAILADDRESS), LOCALIZE (L"Registration_EmailAddress"));
  66.       if (options.registration.user_email[0] != 0)
  67.          SetDlgItemText (hWnd, EDITBOX_REGISTRATION_EMAILADDRESS, options.registration.user_email);
  68.       else
  69.          SetDlgItemText (hWnd, EDITBOX_REGISTRATION_EMAILADDRESS, LOCALIZE (L"Registration_YourEmailHere"));
  70.       SetWindowText (GetDlgItem (hWnd, BUTTON_VALIDATECODE), LOCALIZE (L"Registration_ValidateCode"));
  71.       Static_SetText (GetDlgItem (hWnd, STATICTEXT_REGISTRATION_STATUSBAR), LOCALIZE (L"Registration_StatusBar"));
  72.  
  73.       // set focus to the first settable item
  74.       SendMessage (GetDlgItem (hWnd, EDITBOX_REGISTRATION_EMAILADDRESS), EM_SETSEL, 0, -1); // select all text
  75.       SetFocus (GetDlgItem (hWnd, EDITBOX_REGISTRATION_EMAILADDRESS));
  76.  
  77.       // convert the Paypal bitmap and the status bar message to hyperlinks
  78.       ConvertStaticToHyperlink (GetDlgItem (hWnd, BUTTON_DONATE));
  79.       ConvertStaticToHyperlink (GetDlgItem (hWnd, STATICTEXT_REGISTRATION_STATUSBAR));
  80.    }
  81.  
  82.    // else did we click the close button on the title bar ?
  83.    else if (message == WM_CLOSE)
  84.       EndDialog (hWnd, 0); // close the dialog box
  85.  
  86.    // else did we take action on one of the controls ?
  87.    else if (message == WM_COMMAND)
  88.    {
  89.       // did we cancel the dialog box ? (IDCANCEL is a system-wide dialog box handler value, that catches the ESC key)
  90.       if (wParam_loword == IDCANCEL)
  91.          EndDialog (hWnd, 0); // close the dialog box
  92.  
  93.       // else did the user enter something in the edit boxes ?
  94.       else if (wParam_hiword == EN_CHANGE)
  95.       {
  96.          // enable the validation button only if the PayPal email field has data
  97.          GetDlgItemText (hWnd, EDITBOX_REGISTRATION_EMAILADDRESS, candidate_email, WCHAR_SIZEOF (candidate_email));
  98.  
  99.          // strip spaces and unprintable characters from candidate email, and bring it lowercase
  100.          length = wcslen (candidate_email);
  101.          write_index = 0;
  102.          for (read_index = 0; read_index < length; read_index++)
  103.             if (!iswspace (candidate_email[read_index]))
  104.             {
  105.                candidate_email[write_index] = tolower (candidate_email[read_index]); // force lowercase
  106.                write_index++; // keep only valid signs and discard spaces
  107.             }
  108.          candidate_email[write_index] = 0; // ensure string is correctly terminated
  109.  
  110.          EnableWindow (GetDlgItem (hWnd, BUTTON_VALIDATECODE), ((candidate_email[0] != 0) && (wcslen (candidate_email) > 5) && (wcscmp (candidate_email, LOCALIZE (L"Registration_YourEmailHere")) != 0)
  111.                                                                 && (wcschr (candidate_email, L'@') != NULL) && (wcschr (candidate_email, L'.') != NULL)));
  112.       }
  113.  
  114.       // else was it the validation button ?
  115.       else if (wParam_loword == BUTTON_VALIDATECODE)
  116.       {
  117.          // query the remote server for the code associated with this email. If it fails, ask why.
  118.          candidate_code = RetrieveActivationCode (candidate_email, &failure_reason);
  119.  
  120.          // was there a problem retrieving the code ?
  121.          if ((failure_reason != NULL) && (failure_reason[0] != 0))
  122.             MessageBox (hWnd, failure_reason, PROGRAM_NAME, MB_ICONERROR | MB_OK); // registration failed. Display an error and DON'T exit the program
  123.  
  124.          // else is the retrieved data correct ?
  125.          else if (IsRegistrationCorrect (candidate_email, candidate_code))
  126.          {
  127.             // if so, save activation email and activation code
  128.             wcscpy_s (options.registration.user_email, WCHAR_SIZEOF (options.registration.user_email), candidate_email);
  129.             options.registration.activation_code = candidate_code;
  130.             MessageBox (hWnd, LOCALIZE (L"Registration_ThankYou"), PROGRAM_NAME, MB_ICONINFORMATION | MB_OK);
  131.             EndDialog (hWnd, 0); // then display a thank you dialog box and exit the program
  132.          }
  133.  
  134.          // else the supplied data is wrong
  135.          else
  136.          {
  137.             swprintf_s (temp_string, WCHAR_SIZEOF (temp_string), LOCALIZE (L"Registration_Error"), AUTHOR_EMAIL);
  138.             MessageBox (hWnd, temp_string, PROGRAM_NAME, MB_ICONERROR | MB_OK); // wrong activation. Display an error and DON'T exit the program
  139.          }
  140.       }
  141.  
  142.       // else was it the PayPal button ?
  143.       else if (wParam_loword == BUTTON_DONATE)
  144.       {
  145.          if (wcscmp (languages[language_id].name, L"French") == 0)
  146.             ShellExecute (NULL, L"open", PAYPAL_URL_FR, NULL, NULL, SW_MAXIMIZE); // open PayPal in French in the default browser, maximized
  147.          else
  148.             ShellExecute (NULL, L"open", PAYPAL_URL_XX, NULL, NULL, SW_MAXIMIZE); // open PayPal in English (default) the default browser, maximized
  149.       }
  150.  
  151.       // else was it the status bar hyperlink ?
  152.       else if (wParam_loword == STATICTEXT_REGISTRATION_STATUSBAR)
  153.          ShellExecute (NULL, L"open", PROGRAM_URL, NULL, NULL, SW_MAXIMIZE); // open the main page in the default browser, maximized
  154.    }
  155.  
  156.    // call the default dialog message processing function to keep things going
  157.    return (false);
  158. }
  159.  
  160.  
  161. static unsigned __int32 RetrieveActivationCode (wchar_t *candidate_email, wchar_t **failure_reason)
  162. {
  163.    // this function queries the remote server for registration status. If OK, the received code is returned.
  164.    // On error, 0 is returned and error_string points to the cause of the error (as reported by the server)
  165.    // which is stored in a static buffer in this very function.
  166.  
  167.    static char http_buffer[1024]; // used both for request and reply
  168.    static char ascii_email[128];
  169.  
  170.    struct sockaddr_in service;
  171.    struct hostent *hostinfo;
  172.    char *data_start;
  173.    WSADATA wsa_data;
  174.    int write_index;
  175.    int read_index;
  176.    int length;
  177.    SOCKET s;
  178.  
  179.    // initialize WinSock
  180.    if (WSAStartup (MAKEWORD (2, 2), &wsa_data) != 0)
  181.    {
  182.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  183.       return (0); // couldn't initialize WinSock, return an error condition
  184.    }
  185.  
  186.    // get our distribution server's IP address from the host name
  187.    hostinfo = gethostbyname ("pmbaty.com");
  188.    if (hostinfo == NULL)
  189.    {
  190.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  191.       return (0); // couldn't resolve hostname, return an error condition
  192.    }
  193.  
  194.    // fill in the sockaddr server structure with the server hostinfo data
  195.    service.sin_family = AF_INET;
  196.    service.sin_addr.s_addr = inet_addr (inet_ntoa (*(struct in_addr *) hostinfo->h_addr_list[0]));
  197.    service.sin_port = htons (80); // connect to webserver there (port 80)
  198.  
  199.    // create our socket
  200.    if ((s = socket (AF_INET, SOCK_STREAM, IPPROTO_TCP)) == INVALID_SOCKET)
  201.    {
  202.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  203.       return (0); // couldn't create socket, return an error condition
  204.    }
  205.  
  206.    // connect to the distributor's webserver using that socket
  207.    if (connect (s, (struct sockaddr *) &service, sizeof (service)) == -1)
  208.    {
  209.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  210.       return (0); // unable to connect to webserver, return an error condition
  211.    }
  212.  
  213.    // build the HTTP query and send it
  214.    ConvertTo7BitASCII (ascii_email, sizeof (ascii_email), candidate_email);
  215.    sprintf_s (http_buffer, sizeof (http_buffer),
  216.       "GET /chess/whatsmycode.php?email=%s HTTP/1.1\r\n"
  217.       "Host: pmbaty.com\r\n"
  218.       "Connection: close\r\n"
  219.       "\r\n", ascii_email);
  220.    length = strlen (http_buffer);
  221.    write_index = send (s, http_buffer, length, 0); // send the HTTP query
  222.    if (write_index != length)
  223.    {
  224.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  225.       return (0); // unable to send HTTP query, return an error condition
  226.    }
  227.  
  228.    // read the reply (10 seconds timeout)
  229.    http_buffer[0] = 0;
  230.    read_index = recv_with_timeout (s, 10.0f, http_buffer, sizeof (http_buffer), 0);
  231.    if (read_index < 1)
  232.    {
  233.       *failure_reason = LOCALIZE (L"Registration_NetworkFailure");
  234.       return (0); // empty or erroneous HTTP reply, return an error condition
  235.    }
  236.  
  237.    closesocket (s); // finished communicating, close TCP socket
  238.    WSACleanup (); // and clean up WinSock
  239.  
  240.    // terminate recv buffer and see where the useful data starts
  241.    http_buffer[read_index] = 0;
  242.    //MessageBoxA (NULL, http_buffer, "HTTP response", MB_OK);
  243.    if ((data_start = strstr (http_buffer, "Success=")) != NULL)
  244.    {
  245.       *failure_reason = L""; // no error, set an empty string as the failure reason
  246.       return ((unsigned __int32) atoll (&data_start[strlen ("Success=")])); // and return the candidate code we got
  247.    }
  248.    else if ((data_start = strstr (http_buffer, "Error=")) != NULL)
  249.    {
  250.       data_start += strlen ("Error=");
  251.       if (strchr (data_start, '\r') != NULL) *strchr (data_start, '\r') = 0;
  252.       if (strchr (data_start, '\n') != NULL) *strchr (data_start, '\n') = 0;
  253.       if      (_stricmp (data_start, "WrongEmail") == 0)         *failure_reason = LOCALIZE (L"Registration_WrongEmail");
  254.       else if (_stricmp (data_start, "UnknownEmail") == 0)       *failure_reason = LOCALIZE (L"Registration_UnknownEmail");
  255.       else if (_stricmp (data_start, "TooManyActivations") == 0) *failure_reason = LOCALIZE (L"Registration_TooManyActivations");;
  256.       return (0); // server returned an error, return an error condition
  257.    }
  258.  
  259.    *failure_reason = LOCALIZE (L"Registration_NetworkFailure"); // this should never happen
  260.    return (0); // server returned an error, return an error condition
  261. }
  262.  
  263.  
  264. static int recv_with_timeout (int socket_id, float timeout_in_seconds, char *outbuf, size_t outbuf_size, int flags)
  265. {
  266.    unsigned long nonblocking_mode;
  267.    unsigned long msec_start;
  268.    float timediff;
  269.    int total_size;
  270.    int recv_size;
  271.  
  272.    // make socket non blocking
  273.    nonblocking_mode = 1;
  274.    ioctlsocket (socket_id, FIONBIO, &nonblocking_mode);
  275.  
  276.    // loop endlessly, noting the time at which we start
  277.    msec_start = GetTickCount ();
  278.    total_size = 0;
  279.    for (;;)
  280.    {
  281.       // see how much time elapsed since the last time we received data
  282.       timediff = (GetTickCount () - msec_start) / 1000.0f;
  283.       if (timediff > timeout_in_seconds)
  284.          break; // if we've waited long enough, give up
  285.  
  286.       // see if we have something to read from the socket
  287.       recv_size = recv (socket_id, &outbuf[total_size], outbuf_size - total_size, flags);
  288.       if (recv_size == 0)
  289.          break; // on TCP disconnection, give up too
  290.       else if (recv_size < 0)
  291.       {
  292.          Sleep (100); // if nothing was received then we want to wait a little before trying again, 0.1 seconds
  293.          continue;
  294.       }
  295.  
  296.       total_size += recv_size; // increase the received bytes count
  297.       outbuf[total_size] = 0; // terminate outbuf ourselves
  298.       if (total_size == outbuf_size)
  299.          break; // if the output buffer is full, give up
  300.       msec_start = GetTickCount (); // and remember when we last received data (i.e. now)
  301.    }
  302.  
  303.    return (total_size); // and return the number of bytes received
  304. }
  305.