#include "precomp.h" // // SWL.CPP // Shared Window List // // Copyright(c) Microsoft 1997- // #define MLZ_FILE_ZONE ZONE_CORE // // SWL strategy when network packets are not available // // The SWL only sends one type of message - the window structure message. // When no network packets are available the SWL will drop its current // packet and remember that the window structure has changed since it was // last able to send a packet. SWL_Periodic will also return FALSE when // this happens so that the DCS will know not to send any updates if it // failed to send a window structure. // // This pending of window structure messages is integrated with the // ignore envelopes where the SWL wants to ignore changes caused by itself // (or other components if they call the SWL_Begin/EndIgnoreWindowChanges // functions). // // // SWL strategy for backward compatibility. // // The differences between the R2.0 and 3.0 SWL protocol are: // 1. Tokenless packets. // 2. No shadows. // // // SWL_PartyLeftShare() // void ASShare::SWL_PartyLeftShare(ASPerson * pasPerson) { DebugEntry(ASShare::SWL_PartyLeftShare); ValidatePerson(pasPerson); // // 2.x nodes will fake up a packet for a remote leaving with an empty // window list. That's how they'd nuke shadows for that person, if he // had been hosting. In so doing, they'd use a new token. We need to // bump our token value up also so that the next window list we send // is not dropped. // m_swlLastTokenSeen = SWL_CalculateNextToken(m_swlLastTokenSeen); TRACE_OUT(("SWL_PartyLeftShare: bumped up token to 0x%08x", m_swlLastTokenSeen)); DebugExitVOID(ASShare::SWL_PartyLeftShare); } // // SWL_SyncOutgoing // void ASHost::SWL_SyncOutgoing(void) { DebugEntry(ASHost::SWL_SyncOutgoing); // // Ensure that we send an SWL packet next time we need. // m_swlfForceSend = TRUE; m_swlfSyncing = TRUE; DebugExitVOID(ASHost::SWL_SyncOutgoing); } // // SWL_HostStarting() // BOOL ASHost::SWL_HostStarting(void) { BOOL rc = FALSE; DebugEntry(ASHost::SWL_HostStarting); // // Get an atom to use in getting and setting window properties (which // will give us SWL information about the window). // m_swlPropAtom = GlobalAddAtom(SWL_ATOM_NAME); if (!m_swlPropAtom) { ERROR_OUT(( "GlobalAddAtom error %#x", GetLastError())); DC_QUIT; } // // If this is NT, get the name of our startup desktop // if (!g_asWin95) { ASSERT(m_aswlOurDesktopName[0] == 0); GetUserObjectInformation(GetThreadDesktop(g_asMainThreadId), UOI_NAME, m_aswlOurDesktopName, sizeof(m_aswlOurDesktopName), NULL); TRACE_OUT(("Our desktop name is %s", m_aswlOurDesktopName)); } if (!m_aswlOurDesktopName[0]) { // Use default name TRACE_OUT(("Couldn't get desktop name; using %s", NAME_DESKTOP_DEFAULT)); lstrcpy(m_aswlOurDesktopName, NAME_DESKTOP_DEFAULT); } // // Allocate memory for the window titles. We fix the maximum size of // window title we will send - task list doesn't scroll horizontally so // we truncate window titles at MAX_WINDOW_TITLE_SEND. However, we do // not pad the titles so we try to send as little data as possible. // Allocate all the segment but the rest of the code does not rely on // this so we split them into more segments later if need be. The // memory pointed to by winNames[0] etc looks like this: // // For each entry in the corresponding WinStruct which is a window from // a shared task (and in the same order): // // either - // (char)0xFF - not a `task window' - give it a NULL title // or - // a null terminated string up to MAX_WINDOW_TITLE_SEND characters // // Note that we don't need full and compact versions because only // windows which will be in the compact WinStruct will have // corresponding entries in this structure. // m_aswlWinNames[0] = new char[2*SWL_MAX_WINDOWS*SWL_MAX_WINDOW_TITLE_SEND]; if (!m_aswlWinNames[0]) { ERROR_OUT(( "failed to get memory for window title lists")); DC_QUIT; } m_aswlWinNames[1] = m_aswlWinNames[0] + SWL_MAX_WINDOWS*SWL_MAX_WINDOW_TITLE_SEND; ASSERT(m_swlCurIndex == 0); rc = TRUE; DC_EXIT_POINT: DebugExitBOOL(ASHost::SWL_HostStarting, rc); return(rc); } // // SWL_HostEnded() // void ASHost::SWL_HostEnded(void) { DebugEntry(ASHost::SWL_HostEnded); // // For 2.x nodes, we must send out one last packet so they kill // their shadows. // // // Remove the SWL properties for all existing windows. // EnumWindows(SWLDestroyWindowProperty, 0); m_swlfSyncing = FALSE; if (m_pShare->m_scShareVersion < CAPS_VERSION_30) { // // SWL_Periodic() should NOT put properties on windows // when we're not hosting anymore. // ASSERT(m_pShare->m_pasLocal->hetCount == 0); TRACE_OUT(("SWL_HostEnded: Must send an empty window list for 2.x nodes")); m_swlfForceSend = TRUE; SWL_Periodic(); } if (m_aswlNRInfo[0]) { delete[] m_aswlNRInfo[0]; m_aswlNRInfo[0] = NULL; } if (m_aswlNRInfo[1]) { delete[] m_aswlNRInfo[1]; m_aswlNRInfo[1] = NULL; } if (m_aswlWinNames[0]) { delete[] m_aswlWinNames[0]; m_aswlWinNames[0] = NULL; } if (m_swlPropAtom) { GlobalDeleteAtom(m_swlPropAtom); m_swlPropAtom = 0; } DebugExitVOID(ASHost::SWL_HostEnded); } // // FUNCTION: SWL_GetSharedIDFromLocalID // // DESCRIPTION: // // Given a window ID from a shared application which is running locally // this will return the top level parent. If this parent is invisible, // we return NULL. // // the parent window nearest the desktop. If this parent window is // invisible NULL is returned. // // PARAMETERS: // // window - the window in question // // RETURNS: // // a HWND or NULL if the window is not from a shared application // // HWND ASHost::SWL_GetSharedIDFromLocalID(HWND window) { HWND hwnd; HWND hwndParent; HWND hwndDesktop; DebugEntry(ASHost::SWL_GetSharedIDFromLocalID); hwnd = window; if (!hwnd) { DC_QUIT; } hwndDesktop = GetDesktopWindow(); // // Get the real top level ancestor // while (GetWindowLong(hwnd, GWL_STYLE) & WS_CHILD) { hwndParent = GetParent(hwnd); if (hwndParent == hwndDesktop) break; hwnd = hwndParent; } // // Is this a hosted guy? // if (m_pShare->HET_WindowIsHosted(hwnd)) { if (!(GetWindowLong(hwnd, GWL_STYLE) & WS_VISIBLE)) { // // This window does not have the visible style. But it may just // be transiently invisible and SWL is still treating it as // visible. RAID3074 requires that a window which is not yet // believed to be invisible by SWL is treated as visible (since // the remote has not been informed that it is invisible). We // can determine whether SWL is traeting this window as visible // by looking at the SWL window property. If no property exists // then the window is new so the remote cannot know about it // and we can assume it is indeed invisible. // if (! ((UINT_PTR)GetProp(hwnd, MAKEINTATOM(m_swlPropAtom)) & SWL_PROP_COUNTDOWN_MASK)) { // // SWL knows that the parent of a shared application is // invisible so we just return NULL. // hwnd = NULL; } } } else { hwnd = NULL; } DC_EXIT_POINT: DebugExitDWORD(ASHost::SWL_GetSharedIDFromLocalID, HandleToUlong(hwnd)); return(hwnd); } // // SWL_UpdateCurrentDesktop() // // This checks what the current desktop is, and if it's changed, updates // the NT input hooks for winlogon/screensaver for the service. But normal // SWL and AWC also make use of this info. // void ASHost::SWL_UpdateCurrentDesktop(void) { HDESK hDeskCurrent = NULL; UINT newCurrentDesktop; char szName[SWL_DESKTOPNAME_MAX]; DebugEntry(ASHost::SWL_UpdateCurrentDesktop); newCurrentDesktop = DESKTOP_OURS; if (g_asWin95) { // Nothing to do DC_QUIT; } // // Get the current desktop. If we can't even get it, assume it's the // winlogon desktop. // hDeskCurrent = OpenInputDesktop(0, TRUE, DESKTOP_READOBJECTS); if (!hDeskCurrent) { TRACE_OUT(("OpenInputDesktop failed; must be WINLOGON")); newCurrentDesktop = DESKTOP_WINLOGON; DC_QUIT; } // Get the name of the current desktop szName[0] = 0; GetUserObjectInformation(hDeskCurrent, UOI_NAME, szName, sizeof(szName), NULL); TRACE_OUT(("GetUserObjectInformation returned %s for name", szName)); if (!lstrcmpi(szName, m_aswlOurDesktopName)) { newCurrentDesktop = DESKTOP_OURS; } else if (!lstrcmpi(szName, NAME_DESKTOP_SCREENSAVER)) { newCurrentDesktop = DESKTOP_SCREENSAVER; } else if (!lstrcmpi(szName, NAME_DESKTOP_WINLOGON)) { newCurrentDesktop = DESKTOP_WINLOGON; } else { newCurrentDesktop = DESKTOP_OTHER; } DC_EXIT_POINT: if (newCurrentDesktop != m_swlCurrentDesktop) { // // If this is the service, adjust where we playback events // and/or block local input. // OSI_DesktopSwitch(m_swlCurrentDesktop, newCurrentDesktop); m_swlCurrentDesktop = newCurrentDesktop; } if (hDeskCurrent != NULL) { CloseDesktop(hDeskCurrent); } DebugExitVOID(ASHost::SWL_UpdateCurrentDesktop); } // // SWL_IsOurDesktopActive() // BOOL ASHost::SWL_IsOurDesktopActive(void) { return(!g_asSharedMemory->fullScreen && (m_swlCurrentDesktop == DESKTOP_OURS)); } // // FUNCTION: SWLInitHostFullWinListEntry // // DESCRIPTION: // // Initializes a hosted window entry in the full window list. // // PARAMETERS: hwnd - Window ID of the hosted window // windowProp - SWL window properties for hwnd // ownerID - Window ID of hwnd's owner // pFullWinEntry - pointer to the list entry to initialize // // RETURNS: Nothing // // void ASHost::SWLInitHostFullWinListEntry ( HWND hwnd, UINT windowProp, HWND hwndOwner, PSWLWINATTRIBUTES pFullWinEntry ) { DebugEntry(ASHost::SWLInitHostFullWinListEntry); // // The window is a shared application hosted locally. // These get the application id, the local window id and the owner // window id. // // Note that the real owner of the window may be a child of a shared // window, and therefore not known to the remote machine. We therefore // pass the real owner to SWL_GetSharedIDFromLocalID() which will // traverse up the owner's window tree until it finds a window that is // shared and store the returned window handle in the window structure. // pFullWinEntry->flags = SWL_FLAG_WINDOW_HOSTED; pFullWinEntry->winID = HandleToUlong(hwnd); pFullWinEntry->extra = GetWindowThreadProcessId(hwnd, NULL); // NOTE: ownerWinID is ignored by NM 3.0 and up. pFullWinEntry->ownerWinID = HandleToUlong(SWL_GetSharedIDFromLocalID(hwndOwner)); // // Check if the window is minimized. // if (IsIconic(hwnd)) { pFullWinEntry->flags |= SWL_FLAG_WINDOW_MINIMIZED; } // // TAGGABLE is for 2.x nodes only; 3.0 and up don't look at this. // if (windowProp & SWL_PROP_TAGGABLE) { pFullWinEntry->flags |= SWL_FLAG_WINDOW_TAGGABLE; } if (windowProp & SWL_PROP_TRANSPARENT) { // // The window is transparent and (to have got this far) must be // shared or the desktop is shared, ie we will be sending the // window but need to fiddle the z-order. Flag the transparency so // we can do the z-order later. // pFullWinEntry->flags |= SWL_FLAG_WINDOW_TRANSPARENT; } else if (GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_TOPMOST) { // // The window is not transparent and is topmost, so set the topmost // flag. // pFullWinEntry->flags |= SWL_FLAG_WINDOW_TOPMOST; } // // If this window is on the task bar then pass this info on // if (windowProp & SWL_PROP_TASKBAR) { pFullWinEntry->flags |= SWL_FLAG_WINDOW_TASKBAR; } else { pFullWinEntry->flags |= SWL_FLAG_WINDOW_NOTASKBAR; } DebugExitVOID(ASHost::SWLInitHostFullWinListEntry); } // // FUNCTION: SWLAddHostWindowTitle // // DESCRIPTION: // // Adds a hosted window title (or blank entry) to our window titles list. // // PARAMETERS: winid - Window ID of the hosted window // windowProp - SWL window properties for winid // ownerID - Window ID of winid's owner // ppWinNames - pointer to pointer to window names structure // // RETURNS: Nothing // // void ASHost::SWLAddHostWindowTitle ( HWND hwnd, UINT windowProp, HWND hwndOwner, LPSTR *ppWinNames ) { int lenTitle; DebugEntry(ASHost::SWLAddHostWindowTitle); // // This window gets an entry in our window titles data if it passes the // following tests // // for Windows: it has no owner, or its owner is invisible // // if ( (windowProp & SWL_PROP_TASKBAR) || hwndOwner == NULL || !IsWindowVisible(hwndOwner) ) { // // LAURABU 2.x COMPAT: // 3.0 nodes only look at the text if TASKBAR is set. When 2.x // compat is gone, don't send text in the other cases. // // // Get the title - truncated and null terminated for us. First // look for the desktop, which may have a special, configurable // name. // lenTitle = GetWindowText(hwnd, *ppWinNames, SWL_MAX_WINDOW_TITLE_SEND); // // Check that the title has been null terminated. // (*ppWinNames)[lenTitle] = '\0'; *ppWinNames += lenTitle; } else { // // This is not a task window - put a corresponding entry in the // title info. // **ppWinNames = '\xff'; } *ppWinNames += 1; DebugExitVOID(ASHost::SWLAddHostWindowTitle); } // // FUNCTION: SWL_InitFullWindowListEntry // // DESCRIPTION: // // Initialises an entry in the full window list. // // PARAMETERS: hwnd - Window ID of the window for which an entry is // initialized // windowProp - SWL window properties for hwnd // ownerID - Window ID of hwnd's owner // pFullWinEntry - pointer to the list entry to initialize // // RETURNS: Nothing // // void ASHost::SWL_InitFullWindowListEntry ( HWND hwnd, UINT windowProp, LPSTR * ppWinNames, PSWLWINATTRIBUTES pFullWinEntry ) { HWND hwndOwner; RECT rect; DebugEntry(ASHost::SWL_InitFullWindowListEntry); if (windowProp & SWL_PROP_HOSTED) { // // The window is a shared application hosted locally. // Set up an entry in our full window structure. // hwndOwner = GetWindow(hwnd, GW_OWNER); SWLInitHostFullWinListEntry(hwnd, windowProp, hwndOwner, pFullWinEntry); SWLAddHostWindowTitle(hwnd, windowProp, hwndOwner, ppWinNames); } else { // // The window is a local (non-shared) application // pFullWinEntry->flags = SWL_FLAG_WINDOW_LOCAL; // // We set the winID here because we may need this info // again later, but we will NULL it out before we send the // protocol packet out because it is not info that the // remote needs // pFullWinEntry->winID = HandleToUlong(hwnd); pFullWinEntry->extra = MCSID_NULL; pFullWinEntry->ownerWinID = 0; } // // Get the position and size of the window, in inclusive // Virtual Desktop coordinates. // GetWindowRect(hwnd, &rect); // // TAGGABLE is for 2.x nodes only // if (IsRectEmpty(&rect)) { pFullWinEntry->flags &= ~SWL_FLAG_WINDOW_TAGGABLE; } else { if (windowProp & SWL_PROP_TAGGABLE) { if (!SWLWindowIsTaggable(hwnd)) pFullWinEntry->flags &= ~SWL_FLAG_WINDOW_TAGGABLE; } } // // Make the rectangle inclusive. // rect.right -= 1; rect.bottom -= 1; TSHR_RECT16_FROM_RECT(&(pFullWinEntry->position), rect); DebugExitVOID(ASHost::SWL_InitFullWindowListEntry); } // // FUNCTION: SWLCompactWindowList // // DESCRIPTION: // // Compacts the full window list into one containng only those windows SWL // needs to send (hosts and any locals overlapping hosts) // // PARAMETERS: numFullListEntries - number of entries in the full window // list. // pFullWinList - pointer to the full window list // pCompactWinList - pointer to the compact window list // // RETURNS: Number of entries copied to the compact window list // // UINT ASHost::SWLCompactWindowList ( UINT numFullListEntries, PSWLWINATTRIBUTES pFullWinList, PSWLWINATTRIBUTES pCompactWinList ) { UINT fullIndex; UINT compactIndex = 0; UINT i; DebugEntry(ASHost::SWLCompactWindowList); // // For each window in the full list... // for ( fullIndex = 0; fullIndex < numFullListEntries; fullIndex++ ) { if (pFullWinList[fullIndex].flags & SWL_FLAG_WINDOW_LOCAL) { // // This is a local window so we need to track it only if it // overlaps a hosted window. Run through the remaining windows // until we either find an overlapped hosted window (meaning we // must track this local window) or reach the end of the list // (meaning we don't need to track this local window). // for ( i = fullIndex + 1; i < numFullListEntries; i++ ) { // // If this window is hosted and intersects the local // window then we need to track the local window. // if ( (pFullWinList[i].flags & SWL_FLAG_WINDOW_HOSTED) && (COM_Rect16sIntersect(&pFullWinList[fullIndex].position, &pFullWinList[i].position))) { // // Copy the local window to the compact array and // break out the inner loop. // TRACE_OUT(("Add local hwnd 0x%08x to list at %u", pFullWinList[fullIndex].winID, compactIndex)); pCompactWinList[compactIndex++] = pFullWinList[fullIndex]; break; } } } else { // // This is a shadow or hosted window so we must track it. // TRACE_OUT(("Add shared hwnd 0x%08x to list at %u", pFullWinList[fullIndex].winID, compactIndex)); pCompactWinList[compactIndex++] = pFullWinList[fullIndex]; } } DebugExitDWORD(ASHost::SWLCompactWindowList, compactIndex); return(compactIndex); } // // FUNCTION: SWLAdjustZOrderForTransparency // // DESCRIPTION: // // Rearranges the window structure z-order to take account of a transparent // window (winID). Must not be called if the transparent entry is the last // in the compact list. // // PARAMETERS: pTransparentListEntry - pointer to the transparent entry // pLastListEntry - pointer to the last compact window list // entry // winPosition - position of window in names array // pWinNames - hosted window names // sizeWinNames - number of bytes in winNames // // RETURNS: Nothing. // // void ASHost::SWLAdjustZOrderForTransparency ( PSWLWINATTRIBUTES pTransparentListEntry, PSWLWINATTRIBUTES pLastListEntry, UINT winPosition, LPSTR pWinNames, UINT sizeWinNames ) { SWLWINATTRIBUTES winCopyBuffer; LPSTR pEndNames = &pWinNames[sizeWinNames - 1]; UINT nameLen; char windowText[TSHR_MAX_PERSON_NAME_LEN + SWL_MAX_WINDOW_TITLE_SEND]; DebugEntry(ASHost::SWLAdjustZOrderForTransparency); // // - turn off the transparent flag (it's not part of the protocol) // - move the window to the end of the structure, ie bottom of the // z-order (unless the desktop is at the bottom, in which case // the window becomes next to bottom). // TRACE_OUT(("Adjust z-order for transparent hwnd 0x%08x position %u", pTransparentListEntry->winID, winPosition)); pTransparentListEntry->flags &= ~SWL_FLAG_WINDOW_TRANSPARENT; winCopyBuffer = *pTransparentListEntry; // // Shuffle the windows after the transparent entry one place toward the // start of the list. // UT_MoveMemory(pTransparentListEntry, &pTransparentListEntry[1], (LPBYTE)pLastListEntry - (LPBYTE)pTransparentListEntry); *pLastListEntry = winCopyBuffer; // // Now rearrange the window names in the same way. First, find the name // for this window. // ASSERT((sizeWinNames != 0)); for ( ;winPosition != 0; winPosition-- ) { if ( *pWinNames == '\xff' ) { // // No name exists for this window, so just advance past the // 0xff placeholder. // TRACE_OUT(("No name for %u", winPosition-1)); pWinNames++; } else { // // A name exists for this window, so skip past all the // characters, including the NULL terminator. // TRACE_OUT(( "Ignore %s", pWinNames)); while ( *pWinNames != '\0' ) { pWinNames++; } } } // // winNames now points to the start of the name for the window being // reordered. // if ( *pWinNames == '\xff' ) { // // This window has no name and simply has an 0xff placeholder in // the name list. Move all the remaining names down by one and add // the 0xff at the end. // TRACE_OUT(("Reorder nameless window")); UT_MoveMemory(pWinNames, pWinNames + 1, pEndNames - pWinNames); *pEndNames = (char)'\xff'; } else { // // Move as many bytes as there are characters in the window name // then tack the name on the end. // TRACE_OUT(("Reorder %s", pWinNames)); lstrcpy(windowText, pWinNames); nameLen = lstrlen(pWinNames); UT_MoveMemory(pWinNames, pWinNames + nameLen + 1, pEndNames - pWinNames - nameLen); lstrcpy(pEndNames - nameLen, windowText); } DebugExitVOID(ASHost::SWLAdjustZOrderForTransparency); } // // SWL_Periodic() // // DESCRIPTION: // // Called periodically. If the window structure has changed (such that it // impacts remote systems) then send a new one if we can. // // PARAMETERS: // // fSend - TRUE if the caller really wants us to try to send the new // structure. // // RETURNS: SWL_RC_ERROR : An error occurred // SWL_RC_SENT : Window structure sent successfully // SWL_RC_NOT_SENT : No need to send window structure // UINT ASHost::SWL_Periodic(void) { UINT fRC = SWL_RC_NOT_SENT; UINT newIndex; PSWLWINATTRIBUTES newFullWinStruct; PSWLWINATTRIBUTES curFullWinStruct; PSWLWINATTRIBUTES newCompactWinStruct; PSWLWINATTRIBUTES curCompactWinStruct; UINT i; UINT k; BOOL fNoTitlesChanged; HWND hwnd; SWLENUMSTRUCT swlEnumStruct; int complexity; UINT cNonRectData; UINT size; UINT ourSize; HRGN hrgnNR; HRGN hrgnRect; LPRGNDATA pRgnData = NULL; LPTSHR_INT16 pOurRgnData = NULL; LPTSHR_INT16 pEndRgnData; LPTSHR_INT16 pAllocRgnData = NULL; BOOL fNonRectangularInfoChanged; BOOL rgnOK; RECT rectBound; int left; int top; int right; int bottom; int lastleft; int lasttop; int lastright; int lastbottom; int deltaleft; int deltatop; int deltaright; int deltabottom; int lastdeltaleft; int lastdeltatop; int lastdeltaright; int lastdeltabottom; UINT numCompactWins; UINT lastTransparency; UINT winFlags; UINT iHosted; DebugEntry(ASSHost::SWL_Periodic); SWL_UpdateCurrentDesktop(); // // If this party isn't hosting apps (and isn't faking up an empty // packet for 2.x nodes), there's nothing to do. // if (m_pShare->m_pasLocal->hetCount == HET_DESKTOPSHARED) { m_swlfForceSend = FALSE; fRC = SWL_RC_NOT_SENT; DC_QUIT; } // // Get the window structure into the "new" array. // newIndex = (m_swlCurIndex+1)%2; curFullWinStruct = &(m_aswlFullWinStructs[m_swlCurIndex * SWL_MAX_WINDOWS]); newFullWinStruct = &(m_aswlFullWinStructs[newIndex * SWL_MAX_WINDOWS]); // // Free any previously allocated data. // if (m_aswlNRInfo[newIndex]) { delete[] m_aswlNRInfo[newIndex]; m_aswlNRInfo[newIndex] = NULL; } m_aswlNRSize[newIndex] = 0; // // Start from the first child of the desktop - should be the top // top-level window // ZeroMemory(&swlEnumStruct, sizeof(swlEnumStruct)); swlEnumStruct.pHost = this; swlEnumStruct.newWinNames = m_aswlWinNames[newIndex]; swlEnumStruct.newFullWinStruct = newFullWinStruct; // // Before we consider the windows on the windows desktop we check for // an active full-screen session. If there is one then we insert a // local window the size of the physical screen first so that all // applications which are hosted on this system will become obscured // on the remote system. // ASSERT(swlEnumStruct.count == 0); if (!SWL_IsOurDesktopActive()) { newFullWinStruct[0].flags = SWL_FLAG_WINDOW_LOCAL; newFullWinStruct[0].winID = 0; newFullWinStruct[0].extra = MCSID_NULL; newFullWinStruct[0].ownerWinID = 0; newFullWinStruct[0].position.left = 0; newFullWinStruct[0].position.top = 0; newFullWinStruct[0].position.right = (TSHR_UINT16)(m_pShare->m_pasLocal->cpcCaps.screen.capsScreenWidth-1); newFullWinStruct[0].position.bottom = (TSHR_UINT16)(m_pShare->m_pasLocal->cpcCaps.screen.capsScreenHeight-1); swlEnumStruct.count++; } EnumWindows(SWLEnumProc, (LPARAM)&swlEnumStruct); // // Check if we should bail out because of visibility detection // if (swlEnumStruct.fBailOut) { TRACE_OUT(("SWL_MaybeSendWindowList: bailing out due to visibility detection")); fRC = SWL_RC_ERROR; DC_QUIT; } m_aswlWinNamesSize[newIndex] = (UINT)(swlEnumStruct.newWinNames - m_aswlWinNames[newIndex]); m_aswlNumFullWins[newIndex] = swlEnumStruct.count; // // Check whether we found a transparent window. // lastTransparency = swlEnumStruct.count - 1; k = 0; iHosted = 0; while ( (swlEnumStruct.transparentCount > 0) && (k < lastTransparency) ) { // // If the transparent flag is set then rearrange the z-order, // providing the transparent window is not already at the // bottom of the z-order. // if (newFullWinStruct[k].flags & SWL_FLAG_WINDOW_TRANSPARENT) { // // Now continue with the non-rectangular check - but this will // be on the window "shunted down" from what was the next // position in newCompactWinStruct, ie same value of i. We will // see the moved (transparent) window when we reach it // again at the end of this for-loop (when it will have the // transparent flag off, so we don't redo this bit). // SWLAdjustZOrderForTransparency( &newFullWinStruct[k], &newFullWinStruct[lastTransparency], iHosted, m_aswlWinNames[newIndex], m_aswlWinNamesSize[newIndex]); swlEnumStruct.transparentCount--; } else { if (newFullWinStruct[k].flags & SWL_FLAG_WINDOW_HOSTED) { iHosted++; } k++; } } // // Compare the current and new information - if they are identical then // we can quit now. // fNoTitlesChanged = ((m_aswlWinNamesSize[0] == m_aswlWinNamesSize[1]) && (memcmp(m_aswlWinNames[0], m_aswlWinNames[1], m_aswlWinNamesSize[0]) == 0)); if ( fNoTitlesChanged && !m_swlfRegionalChanges && (m_aswlNumFullWins[0] == m_aswlNumFullWins[1]) && (memcmp(newFullWinStruct, curFullWinStruct, (m_aswlNumFullWins[0] * sizeof(SWLWINATTRIBUTES))) == 0) ) { // // We don't need to send a window structure if nothing has changed // unless there has been a send override. // if (m_swlfForceSend) { // // This is a normal call AND there are pending changes. // TRACE_OUT(( "NORMAL, pending changes - send")); if (SWLSendPacket(&(m_aswlCompactWinStructs[m_swlCurIndex * SWL_MAX_WINDOWS]), m_aswlNumCompactWins[m_swlCurIndex], m_aswlWinNames[m_swlCurIndex], m_aswlWinNamesSize[m_swlCurIndex], m_aswlNRSize[m_swlCurIndex], m_aswlNRInfo[m_swlCurIndex]) ) { // // Successfully sent this so reset the m_swlfForceSend // flag. // m_swlfForceSend = FALSE; fRC = SWL_RC_SENT; } else { // // Failed to send this packet so don't reset // m_swlfForceSend so that we retry next time and return // an error. // fRC = SWL_RC_ERROR; } } else { // // This is a normal call and we don't have any changes pending // so don't send anything. // TRACE_OUT(( "No changes - SWL not sent")); } DC_QUIT; } // // We can reset the flag that alerted us to potential regional window // changes now that we have gone and actually checked all the windows. // m_swlfRegionalChanges = FALSE; // // Something in the window structure has changed. Determine which // windows in the full list are unnecessary (local ones not overlapping // any hosted ones) and create a compact array of windows we really // need. // curCompactWinStruct = &(m_aswlCompactWinStructs[m_swlCurIndex * SWL_MAX_WINDOWS]); newCompactWinStruct = &(m_aswlCompactWinStructs[newIndex * SWL_MAX_WINDOWS]); numCompactWins = SWLCompactWindowList(m_aswlNumFullWins[newIndex], newFullWinStruct, newCompactWinStruct); m_aswlNumCompactWins[newIndex] = numCompactWins; // // Run through the compact window list to check for regional windows // cNonRectData = 0; hrgnNR = CreateRectRgn(0, 0, 0, 0); for (i = 0; i < numCompactWins; i++) { winFlags = newCompactWinStruct[i].flags; hwnd = (HWND)newCompactWinStruct[i].winID; // // There are some "fake" windows for which we do not provide a // winID - these will never be non-rectangular anyway. // if ( (hwnd != NULL) && (winFlags & (SWL_FLAG_WINDOW_LOCAL | SWL_FLAG_WINDOW_HOSTED)) ) { // // If any of the remote systems care, see if this window has a // non rectangular region selected into it. // if (GetWindowRgn(hwnd, hrgnNR) != ERROR) { TRACE_OUT(("Regional window 0x%08x", hwnd)); // // There is a region selected in. // // This region is exactly as the application passed it to // Windows, and has not yet been clipped to the window // rectangle itself. // THE COORDS ARE INCLUSIVE, SO WE ADD ONE to BOTTOM-RIGHT // hrgnRect = CreateRectRgn(0, 0, newCompactWinStruct[i].position.right - newCompactWinStruct[i].position.left + 1, newCompactWinStruct[i].position.bottom - newCompactWinStruct[i].position.top + 1); complexity = IntersectRgn(hrgnNR, hrgnNR, hrgnRect); DeleteRgn(hrgnRect); if (complexity == COMPLEXREGION) { // // The intersection is still a non-rectangular region. // // See how big a buffer we need to get the data for // this region. // size = GetRegionData(hrgnNR, 0, NULL); // // The size we are returned is the size of a full // RGNDATAHEADER plus the rectangles stored in DWORDS. // We can get away with just a WORD as the count of the // rectangles, plus using WORDs for each of the // coordinates. // size = (size - sizeof(RGNDATAHEADER)) / 2 + 2; // Max UINT16 check if ((size <= SWL_MAX_NONRECT_SIZE) && (size + cNonRectData < 65535)) { // // We will be able to query this data later, so // we can flag this as a non-rectangular window. // newCompactWinStruct[i].flags |= SWL_FLAG_WINDOW_NONRECTANGLE; cNonRectData += size; TRACE_OUT(("Regional window region is %d bytes", size)); } else { // // This region is far too complex for us, so we // pretend it is simple so we just consider its // bounding box. // TRACE_OUT(("Region too big %d - use bounds", size)); complexity = SIMPLEREGION; } } if (complexity == SIMPLEREGION) { // // The resultant intersection region happens to be a // rectangle so we can send this via the standard // structure. // // Apply the virtual desktop adjustment, make it // inclusive, and remember we were passed back window // relative coords for the region. // TRACE_OUT(( "rectangular clipped regional window")); // Since we are modifying the compact window struct here // we need to call this so we don't falsely assume that // there are no changes in the window struct based on // comparisons of the old and new full window structs m_swlfRegionalChanges = TRUE; GetRgnBox(hrgnNR, &rectBound); newCompactWinStruct[i].position.left = (TSHR_INT16) (newCompactWinStruct[i].position.left + rectBound.left); newCompactWinStruct[i].position.top = (TSHR_INT16) (newCompactWinStruct[i].position.top + rectBound.top); newCompactWinStruct[i].position.right = (TSHR_INT16) (newCompactWinStruct[i].position.left + rectBound.right - rectBound.left - 1); newCompactWinStruct[i].position.bottom = (TSHR_INT16) (newCompactWinStruct[i].position.top + rectBound.bottom - rectBound.top - 1); } } } } // // Get any non-rectangular areas we need. // if (cNonRectData) { // // There was some data needed - allocate some memory for it. // rgnOK = FALSE; pAllocRgnData = (LPTSHR_INT16) new BYTE[cNonRectData]; if (pAllocRgnData) { pOurRgnData = pAllocRgnData; pEndRgnData = (LPTSHR_INT16)((LPBYTE)pAllocRgnData + cNonRectData); rgnOK = TRUE; // // Loop through the windows again, getting the data this time. // for ( i = 0; i < numCompactWins; i++ ) { if (newCompactWinStruct[i].flags & SWL_FLAG_WINDOW_NONRECTANGLE) { GetWindowRgn((HWND)newCompactWinStruct[i].winID, hrgnNR); // // Clip the region to the window once again. // THE COORDS ARE INCLUSIVE, SO ADD ONE TO BOTTOM-RIGHT // hrgnRect = CreateRectRgn(0, 0, newCompactWinStruct[i].position.right - newCompactWinStruct[i].position.left + 1, newCompactWinStruct[i].position.bottom - newCompactWinStruct[i].position.top + 1); IntersectRgn(hrgnNR, hrgnNR, hrgnRect); DeleteRgn(hrgnRect); // // Get the clipped region data. // // We have already excluded windows above that will // return too large a size here, so we know we are only // working with reasonable sizes now. // size = GetRegionData(hrgnNR, 0, NULL); // // For the moment we allocate memory each time for the // region. Perhaps a better idea would be to save the // max size from when we previously queried the region // sizes, and allocate just that size one outside the // loop. // pRgnData = (LPRGNDATA) new BYTE[size]; if (pRgnData) { GetRegionData(hrgnNR, size, pRgnData); // // There is a possibility that regions will have // changed since we calculated the amount of data // required. Before updating our structure with // this window's region, check // - the window hasn't become normal (ie 0 rects) // - there is still enough space for the rects. // // // Make sure this window still has regions // if (pRgnData->rdh.nCount == 0) { WARNING_OUT(( "No rects for window %#x", newCompactWinStruct[i].winID)); newCompactWinStruct[i].flags &= ~SWL_FLAG_WINDOW_NONRECTANGLE; delete[] pRgnData; // // Move on to next window. // continue; } // // Check we have enough space for the rects: // - ourSize is the number of int16s required. // - GetRegionData returns the number of // rectangles. // // We need one extra int16 to contain the count of // rectangles. // ourSize = (pRgnData->rdh.nCount * 4) + 1; if ((pOurRgnData + ourSize) > pEndRgnData) { WARNING_OUT(( "Can't fit %d int16s of region data", ourSize)); rgnOK = FALSE; delete[] pRgnData; // // Give up processing regional windows. // break; } // // Copy the data across to our SWL area in a more // compact form. // // We take care to produce a compressible form // because the raw data is essentially // uncompressible via sliding window techniques. // (Basically boils down to trying hard to make // most values 0, or else of small magnitude). // // // First we write the count of the number of // rectangles. // *pOurRgnData++ = LOWORD(pRgnData->rdh.nCount); // // Now store the encoded rectangles. // lastleft = 0; lasttop = 0; lastright = 0; lastbottom = 0; lastdeltaleft = 0; lastdeltatop = 0; lastdeltaright = 0; lastdeltabottom = 0; for ( k = 0; k < (UINT)pRgnData->rdh.nCount; k++ ) { // // Extract 16bit quantities from the data we // were returned. // // We also use inclusive coords whereas Windows // gives us exclusive coords. // left = LOWORD(((LPRECT)(pRgnData-> Buffer))[k].left); top = LOWORD(((LPRECT)(pRgnData-> Buffer))[k].top); right = LOWORD(((LPRECT)(pRgnData-> Buffer))[k].right) - 1; bottom = LOWORD(((LPRECT)(pRgnData-> Buffer))[k].bottom) - 1; // // The rectangles are ordered top to bottom, // left to right, so the deltas are of smaller // magnitude than the values themselves. // deltaleft = left - lastleft; deltatop = top - lasttop; deltaright = right - lastright; deltabottom = bottom - lastbottom; // // In general, the left and right edges are // connected lines, and the rectangles are of // equal height so top/bottom are regular. // // Thus the values form a series which we can // exploit to give a more compressible form. // // We already have the delta in each component, // and these values themselves also form a // series. For a straight line series all the // deltas will be the same, so the "delta in // the delta" will be zero. For a curve, // although not all the deltas are the same, // the "delta in the delta" is probably very // small. // // A set of lots of zeros and small magnitude // numbers is very compressible. // // Thus we store the "delta in the delta" for // all components, rather than the values // themselves. The receiver can undo all the // deltaing to arive back at the original // values. // *pOurRgnData++ = (TSHR_UINT16)(deltaleft - lastdeltaleft); *pOurRgnData++ = (TSHR_UINT16)(deltatop - lastdeltatop); *pOurRgnData++ = (TSHR_UINT16)(deltaright - lastdeltaright); *pOurRgnData++ = (TSHR_UINT16)(deltabottom - lastdeltabottom); // // Update our last values. // lastleft = left; lasttop = top; lastright = right; lastbottom = bottom; lastdeltaleft = deltaleft; lastdeltatop = deltatop; lastdeltaright = deltaright; lastdeltabottom = deltabottom; } // // Free the data now we are finished with it. // delete[] pRgnData; } else { // // Failed to get memory for the rectangles, so the // best we can do is use the bounding rect // // Clear the nonrect flag. // TRACE_OUT(("Failed alloc %d - use bounds", i)); newCompactWinStruct[i].flags &= ~SWL_FLAG_WINDOW_NONRECTANGLE; } if (newCompactWinStruct[i].flags & SWL_FLAG_WINDOW_LOCAL) { // // The protocol defines that we will send a NULL // winID for local windows, so NULL it out, now // that we have finished with it. // newCompactWinStruct[i].winID = 0; } } } } if (!rgnOK) { // // Something went wrong, one of: // - we failed to allocate the memory we need to store the // non-rectangular data // - we allocated the memory but it turned out not to be large // enough. // // Either way, best to act as if there is no such data for us. // if (pAllocRgnData == NULL) { WARNING_OUT(( "Failed to alloc %d for NRInfo", cNonRectData)); } else { delete[] pAllocRgnData; pAllocRgnData = NULL; } cNonRectData = 0; // // Clear all the nonrect flags since we will not be sending any // data. // for ( i = 0; i < numCompactWins; i++) { newCompactWinStruct[i].flags &= ~SWL_FLAG_WINDOW_NONRECTANGLE; } } } // // Store the NR information // m_aswlNRSize[newIndex] = cNonRectData; m_aswlNRInfo[newIndex] = (LPTSHR_UINT16)pAllocRgnData; // // We have finished with the region now. // DeleteRgn(hrgnNR); // // Did the data we stored change from the last time? // fNonRectangularInfoChanged = ((m_aswlNRSize[0] != m_aswlNRSize[1]) || (memcmp(m_aswlNRInfo[0], m_aswlNRInfo[1], m_aswlNRSize[0]))); TRACE_OUT(("Non-rectinfo changed %d", fNonRectangularInfoChanged)); // // Check again for no changes - quit if we can. // if (fNoTitlesChanged && !fNonRectangularInfoChanged && (m_aswlNumCompactWins[0] == m_aswlNumCompactWins[1]) && (!memcmp(newCompactWinStruct, curCompactWinStruct, (numCompactWins*sizeof(SWLWINATTRIBUTES))))) { if (!m_swlfForceSend) { // // This is a normal call and we don't have any changes pending // so don't send anything. // TRACE_OUT(("NORMAL no changes, not sent")); } else { // // This is a normal call AND there are pending changes. // TRACE_OUT(( "NORMAL pending changes, send")); if (SWLSendPacket(&(m_aswlCompactWinStructs[m_swlCurIndex * SWL_MAX_WINDOWS]), m_aswlNumCompactWins[m_swlCurIndex], m_aswlWinNames[m_swlCurIndex], m_aswlWinNamesSize[m_swlCurIndex], m_aswlNRSize[m_swlCurIndex], m_aswlNRInfo[m_swlCurIndex]) ) { // // Succesfully sent this so reset the m_swlfForceSend // flag. // m_swlfForceSend = FALSE; fRC = SWL_RC_SENT; } else { // // Failed to send this packet so don't reset // m_swlfForceSend so that we retry next time and return // an error. // fRC = SWL_RC_ERROR; } } // // We can exit here with a changed full window structure but an // unchanged compact window structure. By updating the current // index we avoid having to compact the window structure next time // if the full list doesn't change, ie we will exit on the full // list comparison. If the compact structure subsequently changes // then the full structure must also change, so we will detect this // change. // m_swlCurIndex = newIndex; DC_QUIT; } // // Now the window structure has changed so decide what to do. // m_swlCurIndex = newIndex; // // The window structure has changed so try to send it. // if (SWLSendPacket(&(m_aswlCompactWinStructs[m_swlCurIndex * SWL_MAX_WINDOWS]), m_aswlNumCompactWins[m_swlCurIndex], m_aswlWinNames[m_swlCurIndex], m_aswlWinNamesSize[m_swlCurIndex], m_aswlNRSize[m_swlCurIndex], m_aswlNRInfo[m_swlCurIndex]) ) { // // We have succesfully sent changes so reset the m_swlfForceSend // flag. // m_swlfForceSend = FALSE; fRC = SWL_RC_SENT; } else { // // There were changes but we have failed to send them - set the // m_swlfForceSend flag and return error. // We must tell DCS scheduling that we need a callback BEFORE any // more changes are sent out. // m_swlfForceSend = TRUE; fRC = SWL_RC_ERROR; } DC_EXIT_POINT: DebugExitDWORD(ASHost::SWL_Periodic, fRC); return(fRC); } // // SWLEnumProc() // Callback for top level window enumeration // BOOL CALLBACK SWLEnumProc(HWND hwnd, LPARAM lParam) { PSWLENUMSTRUCT pswlEnum = (PSWLENUMSTRUCT)lParam; UINT_PTR property; UINT windowProp; UINT storedWindowProp; UINT visibleCount; BOOL fVisible; BOOL rc = TRUE; DebugEntry(SWLEnumProc); // // FIRST, WE DETERMINE THE PROPERTIES FOR THE WINDOW. // Get the SWL properties for this window. // windowProp = (UINT)pswlEnum->pHost->SWL_GetWindowProperty(hwnd); // // We'll modify windowProp as we go, so keep a copy of the original // value as stored in the window as we may need it later. // storedWindowProp = windowProp; // // HET tracks whether a window is hosted. Find out now and add this // info to our window properties for convenience. // if (pswlEnum->pHost->m_pShare->HET_WindowIsHosted(hwnd)) { windowProp |= SWL_PROP_HOSTED; } // // Find out whether this window is transparent. // A transparent window overpaints the desktop only, ie it is // overpainted by all other windows. In other words, we can // forget about it (treat it as invisible) unless a toolbar itself // is shared. The MSOffice95 // hidden toolbar is a topmost transparent window (SFR1083). // Add a property flag if transparent. // if (GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_TRANSPARENT) { windowProp |= SWL_PROP_TRANSPARENT; } // // If this window is one that we have identified as generating no // remote shadows, then treat it as being invisible. // fVisible = FALSE; if (IsWindowVisible(hwnd) && !(windowProp & SWL_PROP_IGNORE) && (!(windowProp & SWL_PROP_TRANSPARENT) || (windowProp & SWL_PROP_HOSTED))) { // // SFR1083: if the window is transparent but it is hosted, // we need to send it. In such a case we drop into here to do // the normal visibility processing and will handle // z-order issues later. // // We have been informed that a top level window is visible. // Make sure its visible countdown value is reset. // if ((pswlEnum->pHost->m_pShare->m_pasLocal->hetCount != 0) && ((windowProp & SWL_PROP_COUNTDOWN_MASK) != SWL_BELIEVE_INVISIBLE_COUNT)) { // // We were doing an invisibility countdown for this window // but it has re-visibilized, so reset the counter. // TRACE_OUT(( "Reset visible countdown on hwnd 0x%08x", hwnd)); property = storedWindowProp; property &= ~SWL_PROP_COUNTDOWN_MASK; property |= SWL_BELIEVE_INVISIBLE_COUNT; SetProp(hwnd, SWL_ATOM_NAME, (HANDLE)property); } // // This window is visible // fVisible = TRUE; } else { // // LAURABU BOGUS! // With NM 3.0, who cares? It's only 2.x systems that will kill // then recreate the shadow, causing flicker. // // // We are told that this top level window is invisible. // Check whether we're going to believe it. // Some applications (ie WordPerfect, Freelance Graphics) // upset AS-Shares window structure handling by doing something // like this: // // Make a window invisible // Do some processing which would not normally yield // Make the window visible again // // There is a chance that DC-Share will get scheduled whilst // the window is invisible (because of our cunning scheduling) // and we will think the window is invisible when it is not. // // Also, 32bit tasks that use similar methods (Eg Word95, // Freelance graphics and WM_SETREDRAW messages) may be // interrupted while the window is (temporarily) marked as // invisible. When the CORE is scheduled we may, again, think // that the window is invisible when it is not. // // To overcome this the SWL window property contains a // visibility count, initially set to // SWL_BELIEVE_INVISIBLE_COUNT. Following a visible to // invisible switch, the counter is decremented and only when // it reaches zero does SWL believe that the window is // invisible. The counter is reset when a window is detected as // visible and the counter is not SWL_BELIEVE_INVISIBLE_COUNT. // // This would be fine but there are windows when we mistakenly // pretend that a window which really has become invisible // (rather than one which is transitionally invisible) is // visible. This is exposed by menus and dialog boxes. To // reduce this problem we will never pretend a window is // visible if its class has a CS_SAVEBITS style which should // be the case for windows which are transitionally // visible like menus and dialog boxes. // // SFR1083: always treat a transparent window as invisible // if ( !(windowProp & SWL_PROP_TRANSPARENT) && !(windowProp & SWL_PROP_SAVEBITS) ) { visibleCount = windowProp & SWL_PROP_COUNTDOWN_MASK; if ((visibleCount != 0) && (pswlEnum->pHost->m_pShare->m_pasLocal->hetCount > 0)) { // // We are still treating this window as visible, ie we // are doing a visibilty countdown. Update the count in // the window property. // visibleCount--; property = ~SWL_PROP_COUNTDOWN_MASK & storedWindowProp; property |= visibleCount; TRACE_OUT(( "Decrement visible countdown on window 0x%08x to %d", hwnd, visibleCount)); SetProp(hwnd, SWL_ATOM_NAME, MAKEINTATOM(property)); // // Delay sending of updates since the remote still // has a window structure which includes this window // but it is not on the local screen (so any updates // sent may be for the area where this window was and // the remote will not show them). // pswlEnum->fBailOut = TRUE; rc = FALSE; DC_QUIT; } } } // // Only concerned about visible windows. // if (fVisible) { pswlEnum->pHost->SWL_InitFullWindowListEntry(hwnd, windowProp, &(pswlEnum->newWinNames), &(pswlEnum->newFullWinStruct[pswlEnum->count])); // // If we've added a transparent window then remember this. // if (windowProp & SWL_PROP_TRANSPARENT) { pswlEnum->transparentCount++; } // // Update index // pswlEnum->count++; if (pswlEnum->count == SWL_MAX_WINDOWS) { // // We've reached our limit on # of top level windows, so bail // out. // WARNING_OUT(("SWL_MAX_WINDOWS exceeded")); rc = FALSE; } } DC_EXIT_POINT: DebugExitBOOL(SWLEnumProc, rc); return(rc); } // // SWLSendPacket() // // Called when the shared apps of this node have changed shape/text/position/ // zorder or there have been new windows created/old shared windows destroyed. // We must send these updates out to the remote systems. // // RETURNS: TRUE or FALSE - success of failure. // // BOOL ASHost::SWLSendPacket ( PSWLWINATTRIBUTES pWindows, UINT numWindows, LPSTR pTitles, UINT lenTitles, UINT NRInfoSize, LPTSHR_UINT16 pNRInfo ) { PSWLPACKET pSWLPacket; UINT sizeWindowPkt; UINT i; LPSTR pString; LPBYTE pCopyLocation; UINT cCopySize; SWLPACKETCHUNK chunk; #ifdef _DEBUG UINT sentSize; #endif // _DEBUG DebugEntry(ASHost::SWLSendPacket); if (m_pShare->m_pasLocal->hetCount != 0) { // // This is a real packet, not an empty one // if (!UP_MaybeSendSyncToken()) { // // We needed to send a sync token and couldn't so just return // failure immediately. // TRACE_OUT(( "couldn't send sync token")); return(FALSE); } } // // How big a packet do we need? // sizeWindowPkt = sizeof(SWLPACKET) + (numWindows - 1) * sizeof(SWLWINATTRIBUTES) + lenTitles; // // Add in the size of the regional window information, plus the // size we need for the chunk header. // if (NRInfoSize) { if (lenTitles & 1) { // // We need an extra byte for correct alignment // sizeWindowPkt++; } sizeWindowPkt += NRInfoSize + sizeof(SWLPACKETCHUNK); } // // Allocate a packet for the windows data. // pSWLPacket = (PSWLPACKET)m_pShare->SC_AllocPkt(PROT_STR_UPDATES, g_s20BroadcastID, sizeWindowPkt); if (!pSWLPacket) { WARNING_OUT(("Failed to alloc SWL packet, size %u", sizeWindowPkt)); return(FALSE); } // // Packet successfully allocated. Fill in the data and send it. // pSWLPacket->header.data.dataType = DT_SWL; pSWLPacket->msg = SWL_MSG_WINSTRUCT; pSWLPacket->flags = 0; if (m_swlfSyncing) { pSWLPacket->flags |= SWL_FLAG_STATE_SYNCING; m_swlfSyncing = FALSE; } pSWLPacket->numWindows = (TSHR_UINT16)numWindows; pCopyLocation = (LPBYTE)pSWLPacket->aWindows; cCopySize = numWindows*sizeof(SWLWINATTRIBUTES); memcpy(pCopyLocation, pWindows, cCopySize); // // Copy the title information // pCopyLocation += cCopySize; cCopySize = lenTitles; memcpy(pCopyLocation, pTitles, cCopySize); // // Copy any non-rectangular window information. // if (NRInfoSize) { pCopyLocation += cCopySize; // // The chunk must be word aligned in the packet // if (lenTitles & 1) { // // An odd number of bytes of window titles has misaligned us, // so write a 0 (compresses best!) to realign the pointer. // *pCopyLocation++ = 0; } // // Write the chunk header // chunk.size = (TSHR_INT16)(NRInfoSize + sizeof(chunk)); chunk.idChunk = SWL_PACKET_ID_NONRECT; cCopySize = sizeof(chunk); memcpy(pCopyLocation, &chunk, cCopySize); // // Now write the variable info itself // pCopyLocation += cCopySize; cCopySize = NRInfoSize; memcpy(pCopyLocation, pNRInfo, cCopySize); TRACE_OUT(("Non rect data length %d",NRInfoSize)); } // // Backwards compatibility. // pSWLPacket->tick = (TSHR_UINT16)GetTickCount(); pSWLPacket->token = m_pShare->SWL_CalculateNextToken(m_pShare->m_swlLastTokenSeen); TRACE_OUT(("Updating m_swlLastTokenSeen to 0x%08x for sent packet", pSWLPacket->token)); m_pShare->m_swlLastTokenSeen = pSWLPacket->token; pSWLPacket->reserved = 0; #ifdef _DEBUG { int iWin; int cWins; PSWLWINATTRIBUTES pSwl; // Trace out the entries pSwl = pSWLPacket->aWindows; cWins = pSWLPacket->numWindows; TRACE_OUT(("SWLSendPacket: Sending packet with %d windows", cWins)); for (iWin = 0; iWin < cWins; iWin++, pSwl++) { TRACE_OUT(("SWLSendPacket: Entry %d", iWin)); TRACE_OUT(("SWLSendPacket: Flags %08x", pSwl->flags)); TRACE_OUT(("SWLSendPacket: Window %08x", pSwl->winID)); TRACE_OUT(("SWLSendPacket: Position {%04d, %04d, %04d, %04d}", pSwl->position.left, pSwl->position.top, pSwl->position.right, pSwl->position.bottom)); } } #endif // _DEBUG // // Send the windows packet on the UPDATE stream. // if (m_pShare->m_scfViewSelf) m_pShare->SWL_ReceivedPacket(m_pShare->m_pasLocal, &pSWLPacket->header); #ifdef _DEBUG sentSize = #endif // _DEBUG m_pShare->DCS_CompressAndSendPacket(PROT_STR_UPDATES, g_s20BroadcastID, &(pSWLPacket->header), sizeWindowPkt); TRACE_OUT(("SWL packet size: %08d, sent %08d", sizeWindowPkt, sentSize)); DebugExitBOOL(ASHost::SWLSendPacket, TRUE); return(TRUE); } // // SWL_CalculateNextToken() // // This calculates the next token to put in an outgoing SWL packet. This is // only looked at by backlevel systems (<= NM 2.1) who treat all incoming // SWL streams in one big messy global fashion. So we need to put something // there, something that won't scare them but ensure that our // packets aren't ignored if at all possible. // TSHR_UINT16 ASShare::SWL_CalculateNextToken(TSHR_UINT16 currentToken) { UINT increment; TSHR_UINT16 newToken; DebugEntry(ASShare::SWL_CalculateNextToken); // // We use the highest priority increment to make sure our packets get // through. But will this cause collisions with other 3.0 sharers? // Try lowest priority if necessary. // increment = SWL_NEW_ZORDER_FAKE_WINDOW_INC; // // Return the new token // newToken = SWL_MAKE_TOKEN( SWL_GET_INDEX(currentToken) + SWL_GET_INCREMENT(currentToken), increment); DebugExitDWORD(ASShare::SWL_CalculateNextToken, newToken); return(newToken); } // // SWL_ReceivedPacket() // // DESCRIPTION: // // Processes a windows structure packet which has been received from the // PR. This defines the position of the shared windows hosted on the // remote system, any obscured regions, and the Z-order relative to the // shared windows hosted locally. // // NOTE: We don't do any token stuff for _incoming_ packets; we never // want to drop them since we aren't zordering anything locally. We are // simply applying the zorder/region/position info to the client area // drawing. // void ASShare::SWL_ReceivedPacket ( ASPerson * pasFrom, PS20DATAPACKET pPacket ) { PSWLPACKET pSWLPacket; UINT i; UINT j; PSWLWINATTRIBUTES wins; UINT numWins; HRGN hrgnShared; HRGN hrgnObscured; HRGN hrgnThisWindow; HRGN hrgnRect; LPTSHR_INT16 pOurRgnData; LPSTR pOurRgnChunk; UINT cNonRectWindows; BOOL viewAnyChanges; DebugEntry(ASShare::SWL_ReceivedPacket); ValidatePerson(pasFrom); pSWLPacket = (PSWLPACKET)pPacket; switch (pSWLPacket->msg) { // // This is the only packet we currently recognize. // case SWL_MSG_WINSTRUCT: break; default: WARNING_OUT(("Unknown SWL packet msg %d from [%d]", pSWLPacket->msg, pasFrom->mcsID)); DC_QUIT; } // // Update the last token we've seen, if it's greater than the last // one we know about. Unlike 2.x, we don't drop this packet if it isn't. // if (pSWLPacket->token > m_swlLastTokenSeen) { TRACE_OUT(("Updating m_swlLastTokenSeen to 0x%08x, received packet from person [%d]", pSWLPacket->token, pasFrom->mcsID)); m_swlLastTokenSeen = pSWLPacket->token; } else if (pasFrom->cpcCaps.general.version < CAPS_VERSION_30) { WARNING_OUT(("Received SWL packet from [%d] with stale token 0x%08x", pasFrom->mcsID, pSWLPacket->token)); } // // Return immediately and ignore this baby if we aren't sharing. Back // level systems may send us a SYNC packet with no windows before we've // shared, and may send us one final SWL packet after we're done // sharing. // if (!pasFrom->m_pView) { WARNING_OUT(("SWL_ReceivedPacket: Ignoring SWL packet from person [%d] not hosting", pasFrom->mcsID)); DC_QUIT; } // // Set up local variables to access the data in the packet // wins = pSWLPacket->aWindows; numWins = pSWLPacket->numWindows; pOurRgnChunk = (LPSTR)wins + numWins*sizeof(SWLWINATTRIBUTES); TRACE_OUT(("SWL_ReceivedPacket: Received packet with %d windows from [%d]", numWins, pasFrom->mcsID)); // // We can't handle more than SWL_MAX_WINDOWS in the packet // BOGUS: // LauraBu -- We should negotiate this (make it a cap) on how many // windows we can handle receiving. Then we have an easy path to // increase this number. // if (numWins > SWL_MAX_WINDOWS) { ERROR_OUT(("SWL_ReceivedPacket: too many windows (%04d) in packet from [%08d]", numWins, pasFrom->mcsID)); DC_QUIT; } cNonRectWindows = 0; // // The first pass over the arriving packet is to count the amount of // region data and to update the window tray. // viewAnyChanges = FALSE; // // This part we process front to back, since that's the order of the // strings and we use them for putting entries on the traybar. // for (i = 0; i < numWins; i++) { // Mask out bogus old bits that aren't OK to process wins[i].flags &= SWL_FLAGS_VALIDPACKET; TRACE_OUT(("SWL_ReceivedPacket: Entry %d", i)); TRACE_OUT(("SWL_ReceivedPacket: Flags %08x", wins[i].flags)); TRACE_OUT(("SWL_ReceivedPacket: Window %08x", wins[i].winID)); TRACE_OUT(("SWL_ReceivedPacket: Position {%04d, %04d, %04d, %04d}", wins[i].position.left, wins[i].position.top, wins[i].position.right, wins[i].position.bottom)); // // NOTE: // 2.x nodes may send us a packet with an entry for a shadow. // Go look up the REAL shadow rect from its host. // // And fix up the SWL packet then. // if (wins[i].flags & SWL_FLAG_WINDOW_SHADOW) { ASPerson * pasRealHost; TRACE_OUT(("SWLReceivedPacket: Entry is 2.x SHADOW for host [%d]", wins[i].extra)); // This must be a back level dude, giving us an empty rect. ASSERT(wins[i].position.left == 0); ASSERT(wins[i].position.top == 0); ASSERT(wins[i].position.right == 0); ASSERT(wins[i].position.bottom == 0); // Find the real host of this window SC_ValidateNetID(wins[i].extra, &pasRealHost); if (pasRealHost != NULL) { int cSwl = 0; PSWLWINATTRIBUTES pSwl = NULL; // Try to find this window's entry if (pasRealHost == m_pasLocal) { // // This was shared by US. We can just use the scratch // arrays we already have. m_swlCurIndex has the last // one we sent out to everybody in the share, so the // info it has is most likely reflected on that 2x // remote. // if (m_pHost != NULL) { cSwl = m_pHost->m_aswlNumCompactWins[m_pHost->m_swlCurIndex]; pSwl = &(m_pHost->m_aswlCompactWinStructs[m_pHost->m_swlCurIndex * SWL_MAX_WINDOWS]); } } else { // // This was shared by somebody else, not us and not // the person who sent this SWL packet. So go use the // last SWL info we received from them. // if (pasRealHost->m_pView) { cSwl = pasRealHost->m_pView->m_swlCount; pSwl = pasRealHost->m_pView->m_aswlLast; } } // // Loop through the window list for the real host to // find the entry--we'll use the last REAL rect we got // for this window. // while (cSwl > 0) { if (wins[i].winID == pSwl->winID) { // Copy the _real_ position into the packet. TRACE_OUT(("SWLReceivedPacket: Using real rect {%04d, %04d, %04d, %04d}", pSwl->position.left, pSwl->position.top, pSwl->position.right, pSwl->position.bottom)); wins[i].position = pSwl->position; break; } cSwl--; pSwl++; } if (cSwl == 0) { ERROR_OUT(("SWLReceivedPacket: Couldn't find real window %08x from host [%d]", wins[i].winID, wins[i].extra)); } } } // // 2.x nodes send us VD coords, not screen coords. But that's what // we display for them, so that's what we save away. Note that this // works even in the 2.x shadow case above. Hosted and shadowed // windows both get moved in a desktop scroll, so they stay in the // same place in the virtual desktop, meaning that the coords sent // from the host stay the same even if the windows move, meaning that // we can use the coords of the real host to get the real shadow // rect. // if (wins[i].flags & SWL_FLAG_WINDOW_HOSTED) { TRACE_OUT(("SWL_ReceivedPacket: Hosted Window 0x%08x", wins[i].winID)); TRACE_OUT(("SWL_ReceivedPacket: Text %s", ((*pOurRgnChunk == '\xff') ? "" : pOurRgnChunk))); TRACE_OUT(("SWL_ReceivedPacket: Flags %08x", wins[i].flags)); TRACE_OUT(("SWL_ReceivedPacket: Owner %08x", wins[i].ownerWinID)); TRACE_OUT(("SWL_ReceivedPacket: Position {%04d, %04d, %04d, %04d}", wins[i].position.left, wins[i].position.top, wins[i].position.right, wins[i].position.bottom)); // // We are stepping through the titles (which get sent from // downlevel systems) which do not contain an // explicit length) so that we can get to the data that follows // if (*pOurRgnChunk == '\xff') { // // This is the title for a non-task window - there is just // a single byte to ignore // pOurRgnChunk++; } else { // // This is the title for a task window - there is a NULL // terminated string to ignore. // if (wins[i].flags & SWL_FLAG_WINDOW_TASKBAR) { if (VIEW_WindowBarUpdateItem(pasFrom, &wins[i], pOurRgnChunk)) { viewAnyChanges = TRUE; } } pOurRgnChunk += lstrlen(pOurRgnChunk)+1; } } if (wins[i].flags & SWL_FLAG_WINDOW_NONRECTANGLE) { // // We need to know how many windows have non rectangular data // provided. // cNonRectWindows++; } } if (cNonRectWindows) { TRACE_OUT(( "%d non-rect windows", cNonRectWindows)); // // The window title data is variable length bytes, so may end with // incorrect alignment. Any data which follows (currently only // non-rectangular windows data) is word aligned. // // So check if offset from beginning of data is not aligned. Note // that the packet may start on an ODD boundary because we get // a pointer to the data directly and don't allocate a copy. // if ((LOWORD(pSWLPacket) & 1) != (LOWORD(pOurRgnChunk) & 1)) { TRACE_OUT(("SWL_ReceivedPacket: Aligning region data")); pOurRgnChunk++; } // // Loop through the tagged chunks that follow until we find the // one we want. // while (((PSWLPACKETCHUNK)pOurRgnChunk)->idChunk != SWL_PACKET_ID_NONRECT) { ERROR_OUT(("SWL_ReceivedPacket: unknown chunk 0x%04x", ((PSWLPACKETCHUNK)pOurRgnChunk)->idChunk)); pOurRgnChunk += ((PSWLPACKETCHUNK)pOurRgnChunk)->size; } TRACE_OUT(("Total non rect data 0x%04x", ((PSWLPACKETCHUNK)pOurRgnChunk)->size)); } // // Now scan the wins array backwards (ie furthest away to closest // window) to calculate the unshared region (obscured or nothing there). // and the shared region. // hrgnShared = CreateRectRgn(0, 0, 0, 0); hrgnObscured = CreateRectRgn(0, 0, 0, 0); // // Create a region we can make use of in the next bit of processing. // hrgnRect = CreateRectRgn(0, 0, 0, 0); hrgnThisWindow = CreateRectRgn(0, 0, 0, 0); // // While we are building the shared/obscured regions, also fill in // the host list. Note that this may contain references to local // windows also if they obscure shared ones. Since we don't reference // the list very often, it's easier to just copy the same stuff. // i = numWins; while (i != 0) { i--; // // Consider whether this is a non rectangular window // if (wins[i].flags & SWL_FLAG_WINDOW_NONRECTANGLE) { UINT numRects; UINT cStepOver; int top; int left; int right; int bottom; int lasttop; int lastleft; int lastright; int lastbottom; int deltaleft; int deltatop; int deltaright; int deltabottom; int lastdeltaleft; int lastdeltatop; int lastdeltaright; int lastdeltabottom; // // A non-rectangular region. We go ahead and create the region // from the rectangles that describe it. // pOurRgnData = (LPTSHR_INT16)(pOurRgnChunk + sizeof(SWLPACKETCHUNK)); // // We need to step through the non-rectangular data because we // are processing windows in reverse z-order. // cStepOver = --cNonRectWindows; while (cStepOver--) { // // The next word in the chain contains the number of // rectangles, so we multiply by 4 to get the number of // words to advance. // pOurRgnData += *pOurRgnData++ * 4; } // // Find the number of rectangles. // numRects = *pOurRgnData++; // // The encoding is based on a series of deltas, based on some // initial assumptions // lastleft = 0; lasttop = 0; lastright = 0; lastbottom = 0; lastdeltaleft = 0; lastdeltatop = 0; lastdeltaright = 0; lastdeltabottom = 0; // // Create the region from the first rectangle. // deltaleft = lastdeltaleft + *pOurRgnData++; deltatop = lastdeltatop + *pOurRgnData++; deltaright = lastdeltaright + *pOurRgnData++; deltabottom = lastdeltabottom + *pOurRgnData++; left = lastleft + deltaleft; top = lasttop + deltatop; right = lastright + deltaright; bottom = lastbottom + deltabottom; // THESE COORDS ARE INCLUSIVE, SO ADD ONE SetRectRgn(hrgnThisWindow, left, top, right+1, bottom+1); while (--numRects > 0) { // // Move to the next rectangle. // lastleft = left; lasttop = top; lastright = right; lastbottom = bottom; lastdeltaleft = deltaleft; lastdeltatop = deltatop; lastdeltaright = deltaright; lastdeltabottom = deltabottom; deltaleft = lastdeltaleft + *pOurRgnData++; deltatop = lastdeltatop + *pOurRgnData++; deltaright = lastdeltaright + *pOurRgnData++; deltabottom = lastdeltabottom + *pOurRgnData++; left = lastleft + deltaleft; top = lasttop + deltatop; right = lastright + deltaright; bottom = lastbottom + deltabottom; // // Get the current rectangle into a region. // THESE COORDS ARE INCLUSIVE SO ADD ONE TO BOTTOM-RIGHT // SetRectRgn(hrgnRect, left, top, right+1, bottom+1); // // Add this region to the combined region. // UnionRgn(hrgnThisWindow, hrgnRect, hrgnThisWindow); } // // Switch from window coords to desktop coords. // OffsetRgn(hrgnThisWindow, wins[i].position.left, wins[i].position.top); } else { // // This window region is simply a rectangle. SetRectRgn(hrgnThisWindow, wins[i].position.left, wins[i].position.top, wins[i].position.right+1, wins[i].position.bottom+1); } // // Update the obscured region. As we are working from the back to // the front of the Z-order we can simply add all local window // entries in the incoming structure and subtract all hosted // windows to arrive at the right answer. // if (wins[i].flags & SWL_FLAG_WINDOW_HOSTED) { // // This is a hosted window, sitting above the previous ones. // Add it to the shared region. // Remove it from the obscured region. // UnionRgn(hrgnShared, hrgnShared, hrgnThisWindow); SubtractRgn(hrgnObscured, hrgnObscured, hrgnThisWindow); } else { // // Local windows // TRACE_OUT(( "Adding window %d (%d,%d):(%d,%d) to obscured rgn", i, wins[i].position.left, wins[i].position.top, wins[i].position.right, wins[i].position.bottom )); // // This is a local window, sitting above the previous ones. // We only care about what part of it intersects the current // shared area of the windows behind it. If it doesn't // intersect the shared area at all, it will add no new // obscured bits. // // So figure out what part of the current shared area is now // obscured. Add that piece to the obscured region, and // subtract it from the shared region. // IntersectRgn(hrgnThisWindow, hrgnShared, hrgnThisWindow); UnionRgn(hrgnObscured, hrgnObscured, hrgnThisWindow); SubtractRgn(hrgnShared, hrgnShared, hrgnThisWindow); } } // // We can destroy the regions we created way back when. // DeleteRgn(hrgnRect); DeleteRgn(hrgnThisWindow); // // Save the new host regions. // // Pass the newly calculated regions to the Shadow Window Presenter. // The view code will take care of repainting the invalid parts. And // will delete what was passed in if not kept. // VIEW_SetHostRegions(pasFrom, hrgnShared, hrgnObscured); // // Save the new window list as the current one. // pasFrom->m_pView->m_swlCount = numWins; memcpy(pasFrom->m_pView->m_aswlLast, wins, numWins * sizeof(SWLWINATTRIBUTES)); // // Finish updating the window list. This will repaint the tray bar. We // do this now instead of earlier so that the visual changes and // window bar changes appear together. // VIEW_WindowBarEndUpdateItems(pasFrom, viewAnyChanges); if ((pSWLPacket->flags & SWL_FLAG_STATE_SYNCING) && (m_scShareVersion < CAPS_VERSION_30)) { // // With 2.x nodes in the picture, we need to do the old 2.x ping- // pongy nonsense. We must force a packet if we're hosting when // we receive a SYNC packet. // if (m_pHost) { m_pHost->m_swlfForceSend = TRUE; } } DC_EXIT_POINT: DebugExitVOID(ASShare::SWL_ReceivedPacket); } // // Name: SWLWindowIsTaggable // // Purpose: Determine if a window would be taggable when hosted // // Returns: TRUE if the window would be taggable // If the window is WS_EX_APPWINDOW or has a caption, it's tagged // // Params: winid - ID of window // // BOOL ASHost::SWLWindowIsTaggable(HWND hwnd) { BOOL rc; DebugEntry(ASHost::SWLWindowIsTaggable); if (GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_APPWINDOW) rc = TRUE; else if ((GetWindowLong(hwnd, GWL_STYLE) & WS_CAPTION) == WS_CAPTION) rc = TRUE; else rc = FALSE; DebugExitBOOL(ASHost::SWLWindowIsTaggable, rc); return(rc); } // // FUNCTION: SWLWindowIsOnTaskBar // // DESCRIPTION: // // Determines whether the given window is represented on the task bar // // PARAMETERS: // // hwnd - window to be queried // // RETURNS: // // TRUE - Window is represented on the task bar // // FALSE - Window is not represented on the task bar // // BOOL ASHost::SWLWindowIsOnTaskBar(HWND hwnd) { BOOL rc = FALSE; HWND owner; RECT rect; DebugEntry(ASHost::SWLWindowIsOnTaskBar); // // Our best understanding as to whether a window is on the task bar is // the following: // // - it is a top level window (has no owner) // AND - it does not have the WS_EX_TOOLWINDOW style // // Oprah1655: Visual Basic apps consist of a visible zero sized window // with no owner and a window owned by the zero sized window. We do // not want the zero sized window to be on the task bar, we do want the // other window to be on the task bar. // // owner = GetWindow(hwnd, GW_OWNER); if (owner == NULL) { if (!(GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_TOOLWINDOW)) { GetWindowRect(hwnd, &rect); if ((rect.left < rect.right) && (rect.top < rect.bottom)) { TRACE_OUT(("window 0x%08x allowed on task bar", hwnd)); rc = TRUE; } else { TRACE_OUT(( "window 0x%08x zero sized", hwnd)); } } } else { // // Is the owner window a top-level window of zero size? // if (GetWindow(owner, GW_OWNER) == NULL) { GetWindowRect(owner, &rect); if (IsRectEmpty(&rect)) { TRACE_OUT(("HWND 0x%08x has zero sized top-level owner", hwnd)); rc = TRUE; } } } DebugExitDWORD(ASHost::SWLWindowIsOnTaskBar, rc); return(rc); } // // SWL_GetWindowProperty() // UINT_PTR ASHost::SWL_GetWindowProperty(HWND hwnd) { UINT_PTR properties; char className[HET_CLASS_NAME_SIZE]; DebugEntry(ASHost::SWL_GetWindowProperty); properties = (UINT_PTR)GetProp(hwnd, MAKEINTATOM(m_swlPropAtom)); if (properties != SWL_PROP_INVALID) DC_QUIT; // // No property for this window - it must be new, so create its // initial property state. // // // Assign an initial value to the property, so we never set a property // of zero (which we reserve to indicate invalid). // properties = SWL_PROP_INITIAL; // // TAGGABLE IS FOR < 3.0 nodes only. // if (SWLWindowIsTaggable(hwnd)) { properties |= SWL_PROP_TAGGABLE; } // // Get all the SWL info which is stored as a window property. // if (SWLWindowIsOnTaskBar(hwnd)) { // // This class of window gets tagged. // properties |= SWL_PROP_TASKBAR; } // // Find out if the window class has the CS_SAVEBITS style. // if (GetClassLong(hwnd, GCL_STYLE) & CS_SAVEBITS) { // // This window's class has the CS_SAVEBITS style. // properties |= SWL_PROP_SAVEBITS; } // // Set the visibility count. This is 0 if the window is currently // invisible, SWL_BELIEVE_INVISIBLE_COUNT if visible. // if (IsWindowVisible(hwnd)) { properties |= SWL_BELIEVE_INVISIBLE_COUNT; } // // Set the window property, which we will retrieve when SWL determines // whether it needs to resend the window structure. // if (m_pShare->m_pasLocal->hetCount > 0) { SetProp(hwnd, SWL_ATOM_NAME, (HANDLE)properties); } DC_EXIT_POINT: DebugExitDWORD(ASHost::SWL_GetWindowProperty, properties); return(properties); } // // FUNCTION: SWLDestroyWindowProperty // // DESCRIPTION: // // Destroys the window property for the supplied window. // // PARMETERS: winID - the window ID of the window for which the property is // destroyed. // // RETURNS: Zero // // BOOL CALLBACK SWLDestroyWindowProperty(HWND hwnd, LPARAM lParam) { // // NOTE LAURABU: // We set the property using a string, which bumps up the ref count, // to work around a Win95 bug. We therefore want to remove it using a // string, which bumps down the ref count. Otherwise we will quickly // get a ref count overflow. // RemoveProp(hwnd, SWL_ATOM_NAME); return(TRUE); }