From: Boris Savelev <boris@altlinux.org>
Date: Fri, 11 Jul 2008 13:50:46 +0000 (+0400)
Subject: fix icons and clipboard (thanks to dimbor)
X-Git-Tag: 3.3.0-alt2~24
X-Git-Url: http://git.etersoft.ru?p=rx%2Fnx.git;a=commitdiff_plain;h=c51ec5f4afcbf6199da5343f52241f6ef8edbbfb

fix icons and clipboard (thanks to dimbor)
---

diff --git a/nxwin/programs/Xserver/hw/nxwin/wincutpaste.c b/nxwin/programs/Xserver/hw/nxwin/wincutpaste.c
index 862f75a..5405568 100644
--- a/nxwin/programs/Xserver/hw/nxwin/wincutpaste.c
+++ b/nxwin/programs/Xserver/hw/nxwin/wincutpaste.c
@@ -78,6 +78,8 @@
 
 #define MIN(a,b) ((a) < (b) ? (a) : (b))
 
+/*#define NXWIN_CLIPBOARD_DEBUG*/
+
 extern          WindowPtr *WindowTable;
 extern          Selection *CurrentSelections;
 extern int      NumCurrentSelections;
@@ -96,6 +98,8 @@ Atom            clientTARGETS;
 Atom            clientTEXT;
 Atom            clientCutProperty;
 Atom            clientCLIPBOARD;
+Atom		UTF8_STRING;
+Atom		COMPOUND_TEXT;
 
 Bool            windowsOwner = FALSE;
 Bool            clientOwner  = FALSE;
@@ -119,7 +123,7 @@ void nxwinClearSelection(void)
      return;
 
 #ifdef NXWIN_CLIPBOARD_DEBUG
-   ErrorF("ClearSelection\n");
+   ErrorF("%d nxwinClearSelection: ClearSelection\n",GetTickCount());
 #endif
 
 for (i = 0; i < MIN(MaxSelections, NumCurrentSelections); i++)
@@ -155,8 +159,16 @@ void nxwinInitSelection(HWND hwnd)
    clientTEXT           = MakeAtom("TEXT", strlen("TEXT"), TRUE);
    clientCutProperty    = MakeAtom("NX_CUT_BUFFER_CLIENT", strlen("NX_CUT_BUFFER_CLIENT"), TRUE);
    clientCLIPBOARD      = MakeAtom("CLIPBOARD", strlen("CLIPBOARD"), TRUE);
+/* */
+   UTF8_STRING		= MakeAtom("UTF8_STRING", strlen("UTF8_STRING"), TRUE);
+   COMPOUND_TEXT        = MakeAtom("COMPOUND_TEXT", strlen("COMPOUND_TEXT"), TRUE);
+/* */
    nxwinClipboardStatus = TRUE;
    windowsOwner         = TRUE;
+#ifdef NXWIN_CLIPBOARD_DEBUG
+/*   ErrorF("%d nxwinInitSelection\n",GetTickCount()); 
+*/
+#endif
 /*   nxwinSetWindowClipboard(" ");*/
 }
 
@@ -165,15 +177,17 @@ void nxwinSetSelectionOwner(Selection *pSelection)
    if (!nxwinClipboardStatus)
      return;
 
-#ifdef NXWIN_CLIPBOARD_DEBUG
-   ErrorF("SetSelectionOwner\n");
-#endif
 
    lastOwnerWindowPtr = pSelection->pWin;
    lastOwnerWindow    = pSelection->window;
    lastOwnerClientPtr = pSelection->client;
    
    nxwinSelection = TRUE;
+
+#ifdef NXWIN_CLIPBOARD_DEBUG
+/*   ErrorF("%d nxwinSetSelectionOwner:  window [%p], pWin [%p]\n",GetTickCount(),pSelection->window,pSelection->pWin);
+*/
+#endif
 /*
     if (pSelection->selection == XA_PRIMARY)
     {
@@ -194,21 +208,67 @@ void nxwinSetSelectionOwner(Selection *pSelection)
                                 NoEventMask, NoEventMask ,
                                 NullGrab);
 #ifdef NXWIN_CLIPBOARD_DEBUG
-       ErrorF("SetSelectionOwner XA_PRIMARY \n");
+       ErrorF(" nxwinSetSelectionOwner: SetSelectionOwner XA_PRIMARY \n");
 #endif
 
        windowsOwner = FALSE;   
      }
 */
+
+}
+
+/* ------------------ My  ------------------------------------------- */
+char * unicode_to_some_cp(wchar_t *unicode_string, int CP)
+{
+	int err;
+	char * res;
+	int res_len = WideCharToMultiByte(
+		CP,				// Code page
+		0,					// Default replacement of illegal chars
+		unicode_string,		// Multibyte characters string
+		-1,					// Number of unicode chars is not known
+		NULL,				// No buffer yet, allocate it later
+		0,					// No buffer
+		NULL,				// Use system default
+		NULL				// We are not interested whether the default char was used
+		);
+	if (res_len == 0) 
+	{
+		return NULL;
+	}
+	res = malloc(res_len);
+	if (res == NULL) 
+	{
+		return NULL;
+	}
+	err = WideCharToMultiByte(
+		CP,				// Code page
+		0,					// Default replacement of illegal chars
+		unicode_string,		// Multibyte characters string
+		-1,					// Number of unicode chars is not known
+		res,				// Output buffer
+		res_len,			// buffer size
+		NULL,				// Use system default
+		NULL				// We are not interested whether the default char was used
+		);
+	if (err == 0)
+	{
+		free(res);
+		return NULL;
+	}
+	return res;
 }
 
+
+/* ------------------ My ------------------------------------------- */
+
 Bool nxwinConvertSelection(ClientPtr client ,WindowPtr pWin, Atom selection, Window requestor, Atom property, Atom target, Time time)
 {
    if (!nxwinClipboardStatus)
      return 0;
 
 #ifdef NXWIN_CLIPBOARD_DEBUG
-   ErrorF("ConvertSelection\n");
+   ErrorF("%d nxwinConvertSelection: pWin [%p], selection [%s], target [%s]\n",GetTickCount(),pWin,NameForAtom(selection),NameForAtom(target));
 #endif
 
    if (!windowsOwner) /* there is a X client owner, let normal stuff happens */
@@ -236,20 +296,25 @@ Bool nxwinConvertSelection(ClientPtr client ,WindowPtr pWin, Atom selection, Win
       return 1;
    }
 
-#ifdef NXWIN_CLIPBOARD_DEBUG
-   ErrorF("ConvertSelection converting...\n");
-#endif
 
-   if ((target == clientTEXT) || (target == XA_STRING))
+   if ((target == clientTEXT) || (target == XA_STRING) || (target == UTF8_STRING))
    {
       HGLOBAL hGlobal;
       char *pszGlobalData;
       xEvent x;
 
+      char *pszData=NULL;
+
       /* Access the clipboard */
       if (!OpenClipboard (lastHwnd)) return 0;
-
-      hGlobal = GetClipboardData (CF_TEXT);
+      if (target == UTF8_STRING)
+      {
+        hGlobal = GetClipboardData (CF_UNICODETEXT);
+      }
+      else
+      {
+        hGlobal = GetClipboardData (CF_TEXT);
+      }
       if (!hGlobal)
       {
         CloseClipboard();
@@ -264,10 +329,29 @@ Bool nxwinConvertSelection(ClientPtr client ,WindowPtr pWin, Atom selection, Win
         return 1;
       }
       pszGlobalData = (char *) GlobalLock (hGlobal);
-
-      /* Convert DOS string to UNIX string */
-      DOStoUNIX (pszGlobalData, strlen (pszGlobalData));
-
+			
+      if (target == UTF8_STRING)
+      {
+         /* Convert DOS string to UNIX string */
+         DOStoUNIX (pszGlobalData, strlen (pszGlobalData));
+         pszData = unicode_to_some_cp((wchar_t *) pszGlobalData,CP_UTF8);
+#ifdef NXWIN_CLIPBOARD_DEBUG
+        ErrorF("  Set sel [%s], target [%s], prop [%s] to UTF-8 string [%s]\n",NameForAtom(selection),NameForAtom(target),NameForAtom(property),pszData);
+#endif
+      /* Copy the clipboard text to the requesting window */
+        ChangeWindowProperty(pWin,
+                             property,
+                             target,
+                             8,
+                             PropModeReplace,
+                             strlen(pszData),
+                             pszData, 1);
+      }
+      else
+      {
+#ifdef NXWIN_CLIPBOARD_DEBUG
+      ErrorF("  Set sel [%s], target [%s], prop [%s] to CP-1251 string [%s]\n",NameForAtom(selection),NameForAtom(target),NameForAtom(property),pszGlobalData);
+#endif
       /* Copy the clipboard text to the requesting window */
       ChangeWindowProperty(pWin,
                            property,
@@ -276,12 +360,16 @@ Bool nxwinConvertSelection(ClientPtr client ,WindowPtr pWin, Atom selection, Win
                            PropModeReplace,
                            strlen(pszGlobalData),
                            pszGlobalData, 1);
+      }
 
       /* Release the clipboard data */
       GlobalUnlock (hGlobal);
       pszGlobalData = NULL;
       CloseClipboard ();
 
+      if (pszData != NULL)
+	free (pszData);
+
       x.u.u.type = SelectionNotify;
       x.u.selectionNotify.time = time;
       x.u.selectionNotify.requestor = requestor;
@@ -370,6 +458,7 @@ UNIXtoDOS (char **ppszData, int iLength)
 }
 
 
+
 void
 DOStoUNIX (char *pszSrc, int iLength)
 {
@@ -401,8 +490,11 @@ void nxwinSetWindowClipboard(char *text, int iLength)
   char          *pszGlobalData;
   char          *pszTemp;
 
+
   if (!nxwinClipboardStatus)
+  {
      return;
+  }
 
   if (OpenClipboard(lastHwnd))
   {
@@ -440,6 +532,9 @@ void nxwinSetWindowClipboard(char *text, int iLength)
       /* free the allocated memory */
       xfree(pszTemp);
 
+#ifdef NXWIN_CLIPBOARD_DEBUG
+   ErrorF("%d nxwinSetWindowClipboard: [%s].\n",GetTickCount(),pszGlobalData);
+#endif
       /* Release the pointer to the global memory */
       GlobalUnlock (hGlobal);
       pszGlobalData = NULL;
@@ -466,10 +561,14 @@ Bool nxwinSendNotify(xEvent* x)
  int iReturn;
 
 #ifdef NXWIN_CLIPBOARD_DEBUG
-  ErrorF("SendNotify\n");
+  ErrorF("%d nxwinSendNotify: pwin [%p];",GetTickCount(),lastOwnerWindowPtr);
+  ErrorF(" x->u.u.type [%s]; x->u.selectionNotify: property [%s]; target [%s]; selection [%s]",NameForAtom(x->u.u.type),NameForAtom(x->u.selectionNotify.property),NameForAtom(x->u.selectionNotify.target),NameForAtom(x->u.selectionNotify.selection)); 
 #endif
   if (x->u.selectionNotify.property == clientCutProperty)
   {
+#ifdef NXWIN_CLIPBOARD_DEBUG
+     ErrorF("\n");
+#endif
      Atom       atomReturnType;
      int     iReturnFormat;
      unsigned long   ulReturnItems;
@@ -479,13 +578,26 @@ Bool nxwinSendNotify(xEvent* x)
                                 AnyPropertyType, &atomReturnType, &iReturnFormat,
                                 &ulReturnItems, &ulReturnBytesLeft, &pszReturnData);
 
+/*
+#ifdef NXWIN_CLIPBOARD_DEBUG
+     ErrorF("nxwinSendNotify: called 1-st GetWindowProperty - status = %d, ulReturnBytesLeft = %d\n",Success,ulReturnBytesLeft);
+#endif
+*/
      if ((iReturn == Success) && (ulReturnBytesLeft > 0))
      {
         iReturn = GetWindowProperty(lastOwnerWindowPtr,clientCutProperty,0,ulReturnBytesLeft, FALSE,
                                         AnyPropertyType, &atomReturnType, &iReturnFormat,
                                         &ulReturnItems, &ulReturnBytesLeft, &pszReturnData);
+/*
+#ifdef NXWIN_CLIPBOARD_DEBUG
+	ErrorF("nxwinSendNotify: called 2-nd GetWindowProperty - status=%d, ulReturnItems=%d\n",Success,ulReturnItems);
+#endif
+*/
         if ((iReturn == Success) && (ulReturnItems > 0))
         {
+#ifdef NXWIN_CLIPBOARD_DEBUG
+  	  ErrorF("  AtomReturnType [%s]\n",NameForAtom(atomReturnType));
+#endif
           nxwinSetWindowClipboard(pszReturnData, ulReturnItems);
 
           clientOwner = TRUE;
@@ -495,6 +607,9 @@ Bool nxwinSendNotify(xEvent* x)
         }
      }
   }
+#ifdef NXWIN_CLIPBOARD_DEBUG
+  else ErrorF(" !! MISSED !!\n");
+#endif
   return FALSE;
 }
 
@@ -512,19 +627,27 @@ void nxwinLostFocus(void)
    if (lastOwnerWindowPtr)
    {
       xEvent x;
+#ifdef NXWIN_CLIPBOARD_DEBUG
+   ErrorF("%d nxwinLostFocus: lastOwnerWindow [%p]; pWin [%p]; target [%s]; prop [%s]\n",GetTickCount(),lastOwnerWindow,lastOwnerWindowPtr,NameForAtom(XA_STRING),NameForAtom(clientCutProperty));
+#endif
 
       x.u.u.type = SelectionRequest;
       x.u.selectionRequest.time = GetTimeInMillis();
       x.u.selectionRequest.owner = lastOwnerWindow;
       x.u.selectionRequest.requestor = WindowTable[0]->drawable.id;
-      x.u.selectionRequest.selection = XA_PRIMARY;
+/* My changes */
+/*
+      x.u.selectionRequest.selection =  XA_PRIMARY; 
+*/
+      x.u.selectionRequest.selection = clientCLIPBOARD;  
+/* My changes */
       x.u.selectionRequest.target = XA_STRING;
       x.u.selectionRequest.property = clientCutProperty;
-
       (void) TryClientEvents (lastOwnerClientPtr, &x, 1,
                                NoEventMask, NoEventMask /* CantBeFiltered */,
                                NullGrab);
       SetCriticalOutputPending();
+
    }
 }
 
diff --git a/nxwin/programs/Xserver/hw/nxwin/winmultiwindowwm.c b/nxwin/programs/Xserver/hw/nxwin/winmultiwindowwm.c
index e8149c7..a09d227 100644
--- a/nxwin/programs/Xserver/hw/nxwin/winmultiwindowwm.c
+++ b/nxwin/programs/Xserver/hw/nxwin/winmultiwindowwm.c
@@ -408,7 +408,9 @@ privateGetWindowName(void *pWin, char **ppName, Atom atom)
    if( (retValue = GetWindowProperty(pWin, atom, 0L, 0L, False,
                         AnyPropertyType, &retType, &retFormat, 
                         &nItems, &bytesLeft, (unsigned char**)NULL)) != Success){
-       ErrorF("GetWindowName: GetWindowProperty failed\n");
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+       ErrorF("GetWindowName: GetWindowProperty /length/ failed\n");
+#endif
        return retValue; 
    }
    origLen = bytesLeft;
@@ -417,7 +419,9 @@ privateGetWindowName(void *pWin, char **ppName, Atom atom)
    if( (retValue = GetWindowProperty(pWin, atom, 0L, bytesLeft, False,
                         AnyPropertyType, &retType, &retFormat, 
                         &nItems, &bytesLeft, (unsigned char**)ppName)) != Success){
-       ErrorF("GetWindowName: GetWindowProperty failed\n");
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+       ErrorF("GetWindowName: GetWindowProperty /value/ failed\n");
+#endif
        free(strName);
        return retValue;
    }
@@ -448,8 +452,16 @@ GetWindowName (void *pWin, char **ppName)
 
    /* TRY with ATOM WM_NAME */
    reqAtom = XA_WM_NAME;
-   if( privateGetWindowName(pWin, ppName, reqAtom) == Success )
+   if( privateGetWindowName(pWin, ppName, reqAtom) == Success ) 
+/* ------------------ My crooked additions ------------------------------------------- */
+/* sometimes *ppName is present and zero length - anyway need to try second atom */
+   { if (strlen(*ppName) > 0)
+     {
       return;
+     }
+     else free(*ppName);
+   }
+/* ------------------ My crooked additions ------------------------------------------- */
    reqAtom = MakeAtom(atom_NET_WM_NAME, sizeof(atom_NET_WM_NAME) - 1, True);
    privateGetWindowName(pWin, ppName, reqAtom);
 }
@@ -537,6 +549,98 @@ winMultStackWindow(pWin, val)
     return 1;
 }
 
+/* ------------------ My crooked additions ------------------------------------------- */
+wchar_t * utf8_to_unicode(char *utf8_string)
+{
+	int err;
+	wchar_t * res;
+	int res_len = 1000;
+	res = malloc(res_len);
+	err = MultiByteToWideChar(
+		CP_UTF8,		// Code page
+		0,			// No flags
+		utf8_string,		// Multibyte characters string
+		-1,			// The string is NULL terminated
+		res,			// Output buffer
+		res_len			// buffer size
+		);
+	if (err == 0)
+	{
+//		printf("Failed to convert to unicode\n");
+		free(res);
+		return NULL;
+	}
+	return res;
+}
+
+void 
+winSetWinName (WMMsgNodePtr pNode)
+{
+  {
+    /*XWindowAttributes		attr;*/
+    char			*pszName;
+    wchar_t 			*pszWName;
+#if 0
+    XWMHints			*pHints;
+#endif
+    /* Get the window attributes */
+    /*
+    XGetWindowAttributes (pWMInfo->pDisplay,
+			  pNode->msg.iWindow,
+			  &attr);
+    */
+    if (!winGetOverrideRedirectPriv(pNode->msg.pWin))
+     {
+#ifdef NXWIN_MULTIWINDOW
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+	if(nxwinMultiwindow)
+             ErrorF("winMultiWindowWMProc: LOCK before GetWindowName\n");
+	else
+	     ErrorF("winMultiWindowWMProc: before GetWindowName\n");	
+#endif
+             if(nxwinMultiwindow && pthread_mutex_lock(&nxwinMultiwindowMutex))
+                     ErrorF("winMultiWindowWMProc: pthread_mutex_lock failed\n");
+#endif
+	/* Set the Windows window name */
+	GetWindowName(pNode->msg.pWin, &pszName);
+#ifdef NXWIN_MULTIWINDOW
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+	if(nxwinMultiwindow)
+             ErrorF("winMultiWindowWMProc: UNLOCK after GetWindowName\n");
+	else
+	     ErrorF("winMultiWindowWMProc: after GetWindowName\n");
+#endif
+        if(nxwinMultiwindow && pthread_mutex_unlock(&nxwinMultiwindowMutex) != 0)
+             ErrorF("winMultiWindowWMProc: !!! pthread_mutex_unlock failed\n");
+#endif
+	if(!pszName){
+	     ErrorF("winMultiWindowWMProc: GetWindowName failed\n");	
+	     return;
+		}
+
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+	ErrorF("winMultiWindowWMProc!!!: Window title before converting - %s\n",pszName);
+#endif
+/* if you know, how to get system charset from server, tell me about them */
+	pszWName = utf8_to_unicode(pszName);
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+	ErrorF("winMultiWindowWMProc!!!: Window title after converting  - %s\n",pszWName);
+#endif
+	SetWindowTextW (pNode->msg.hwndWindow, pszWName); 
+//	SetWindowText (pNode->msg.hwndWindow, pszName);
+
+#ifdef NXWIN_MULTIWINDOW_DEBUG
+	ErrorF("winMultiWindowWMProc: Insert here LoadIcon\n");
+#endif
+	free (pszName);
+        if (pszWName != NULL)
+		free (pszWName);
+      }
+  }
+}
+/* -----------End of my crooked additions ------------------------------------------- */
+
+
 static void *
 winMultiWindowWMProc (void *pArg)
 {
@@ -599,6 +703,10 @@ winMultiWindowWMProc (void *pArg)
      /*
      XRaiseWindow (pWMInfo->pDisplay, pNode->msg.iWindow);
      */
+
+/* ------------------ My crooked additions ------------------------------------------- */
+	  winSetWinName (pNode);
+/* -----------End of my crooked additions ------------------------------------------- */
      break;
 
    case WM_WM_LOWER:
@@ -617,54 +725,9 @@ winMultiWindowWMProc (void *pArg)
 #if CYGMULTIWINDOW_DEBUG
      ErrorF ("\tWM_WM_MAP\n");
 #endif
-	  {
-	    /*XWindowAttributes		attr;*/
-	    char			*pszName;
-#if 0
-	    XWMHints			*pHints;
-#endif
-
-	    /* Get the window attributes */
-	    /*
-	    XGetWindowAttributes (pWMInfo->pDisplay,
-				  pNode->msg.iWindow,
-				  &attr);
-             */
-	    if (!winGetOverrideRedirectPriv(pNode->msg.pWin))
-	      {
-#ifdef NXWIN_MULTIWINDOW
-#ifdef NXWIN_MULTIWINDOW_DEBUG
-		if(nxwinMultiwindow)
-	             ErrorF("winMultiWindowWMProc: LOCK before GetWindowName\n");
-		else
-		     ErrorF("winMultiWindowWMProc: before GetWindowName\n");	
-#endif
-                if(nxwinMultiwindow && pthread_mutex_lock(&nxwinMultiwindowMutex))
-	                     ErrorF("winMultiWindowWMProc: pthread_mutex_lock failed\n");
-#endif
-		/* Set the Windows window name */
-		GetWindowName(pNode->msg.pWin, &pszName);
-#ifdef NXWIN_MULTIWINDOW
-#ifdef NXWIN_MULTIWINDOW_DEBUG
-		if(nxwinMultiwindow)
-	             ErrorF("winMultiWindowWMProc: UNLOCK after GetWindowName\n");
-		else
-		     ErrorF("winMultiWindowWMProc: after GetWindowName\n");
-#endif
-                if(nxwinMultiwindow && pthread_mutex_unlock(&nxwinMultiwindowMutex) != 0)
-	             ErrorF("winMultiWindowWMProc: !!! pthread_mutex_unlock failed\n");
-#endif
-		if(!pszName){
-		     ErrorF("winMultiWindowWMProc: GetWindowName failed\n");	
-		     break;
-		}
-		SetWindowText (pNode->msg.hwndWindow, pszName);
-#ifdef NXWIN_MULTIWINDOW_DEBUG
-		ErrorF("winMultiWindowWMProc: Insert here LoadIcon\n");
-#endif
-		free (pszName);
-	      }
-	  }
+/* ------------------ My crooked additions ------------------------------------------- */
+	  winSetWinName (pNode);
+/* -----------End of my crooked additions ------------------------------------------- */
      break;
 
    case WM_WM_UNMAP:
@@ -904,6 +967,9 @@ winMultiWindowWMProc (void *pArg)
      if(nxwinMultiwindow && pthread_mutex_unlock(&nxwinMultiwindowMutex))
                ErrorF("!!! pthread_mutex_unlock failed\n");
 #endif
+/* ------------------ My crooked additions ------------------------------------------- */
+	  winSetWinName (pNode);
+/* -----------End of my crooked additions ------------------------------------------- */
      break;
 #if 0
    case WM_WM_X_EVENT: