Méthode la plus rapide de capture d'écran sous Windows


159

Je veux écrire un programme de capture d'écran pour la plate-forme Windows, mais je ne sais pas comment capturer l'écran. La seule méthode que je connaisse est d'utiliser GDI, mais je suis curieux de savoir s'il existe d'autres façons de procéder et, s'il y en a, laquelle entraîne le moins de frais généraux? La vitesse est une priorité.

Le programme de screencasting sera pour enregistrer des séquences de jeu, bien que, si cela restreint les options, je suis toujours ouvert à toute autre suggestion qui ne relève pas de cette portée. La connaissance n'est pas mauvaise, après tout.

Edit : Je suis tombé sur cet article: Diverses méthodes pour capturer l'écran . Il m'a présenté la manière de procéder de l'API Windows Media et la manière de procéder par DirectX. Il mentionne dans la conclusion que la désactivation de l'accélération matérielle pourrait considérablement améliorer les performances de l'application de capture. Je suis curieux de savoir pourquoi. Quelqu'un pourrait-il remplir les espaces manquants pour moi?

Edit : J'ai lu que les programmes de screencasting tels que Camtasia utilisent leur propre pilote de capture. Quelqu'un pourrait-il m'expliquer en profondeur comment cela fonctionne et pourquoi c'est plus rapide? J'ai peut-être également besoin de conseils sur la mise en œuvre de quelque chose comme ça, mais je suis sûr qu'il existe de toute façon une documentation existante.

De plus, je sais maintenant comment FRAPS enregistre l'écran. Il accroche l'API graphique sous-jacente pour lire à partir du tampon arrière. D'après ce que je comprends, c'est plus rapide que la lecture à partir du tampon avant, car vous lisez à partir de la RAM système plutôt que de la RAM vidéo. Vous pouvez lire l'article ici .


Avez-vous envisagé, plutôt que d'enregistrer graphiquement le contenu de l'écran, d'utiliser un système de relecture ?
Benjamin Lindley

2
Vous n'avez rien à accrocher. Il vous suffit d'écrire vos événements d'entrée pour qu'ils ne contrôlent pas directement le jeu, mais appellent à la place d'autres fonctions. Par exemple, si le joueur appuie sur la touche gauche, vous ne décrémentez pas simplement la position x du joueur. Au lieu de cela, vous appelez une fonction, comme MovePlayerLeft(). Et vous enregistrez également l'heure et la durée des pressions sur les touches et autres entrées. Ensuite, lorsque vous êtes en mode lecture, vous ignorez simplement l'entrée et lisez à la place les données enregistrées. Si, dans les données, vous voyez une touche gauche, vous appelez MovePlayerLeft().
Benjamin Lindley

1
@PigBen Ce sera une application générique pour enregistrer des séquences de jeu. Ce n'est pas pour un jeu spécifique. Quelqu'un qui appuie sur la touche gauche pourrait signifier aller à droite, pour autant que je sache. De plus, vous n'avez pas pris en compte les événements qui ne sont pas influencés par l'utilisateur. Et qu'en est-il du rendu?
someguy

1
@someguy Ok Je suppose que vous faites quelque chose de beaucoup plus intense, j'avais ajouté une routine en utilisant les méthodes ci-dessus pour enregistrer la relecture des AVI dans un jeu à environ 30 images par seconde sans accroc. J'ai créé un enregistreur d'écran à plusieurs moniteurs en utilisant l'API Windows pour "l'optimisation de la force de travail", mais cela a mal fonctionné même à 4fps ciblés.
AJG85

1
Il y a un pilote miroir open source pour Windows sur le site de dépôt UltraVNC ici ultravnc.svn.sourceforge.net/viewvc/ultravnc/...
Échoué

Réponses:


62

C'est ce que j'utilise pour collecter des images uniques, mais si vous modifiez ceci et gardez les deux cibles ouvertes tout le temps, vous pouvez le «streamer» sur le disque en utilisant un compteur statique pour le nom de fichier. - Je ne me souviens plus où j'ai trouvé ça, mais ça a été modifié, merci à qui que ce soit!

void dump_buffer()
{
   IDirect3DSurface9* pRenderTarget=NULL;
   IDirect3DSurface9* pDestTarget=NULL;
     const char file[] = "Pickture.bmp";
   // sanity checks.
   if (Device == NULL)
      return;

   // get the render target surface.
   HRESULT hr = Device->GetRenderTarget(0, &pRenderTarget);
   // get the current adapter display mode.
   //hr = pDirect3D->GetAdapterDisplayMode(D3DADAPTER_DEFAULT,&d3ddisplaymode);

   // create a destination surface.
   hr = Device->CreateOffscreenPlainSurface(DisplayMde.Width,
                         DisplayMde.Height,
                         DisplayMde.Format,
                         D3DPOOL_SYSTEMMEM,
                         &pDestTarget,
                         NULL);
   //copy the render target to the destination surface.
   hr = Device->GetRenderTargetData(pRenderTarget, pDestTarget);
   //save its contents to a bitmap file.
   hr = D3DXSaveSurfaceToFile(file,
                              D3DXIFF_BMP,
                              pDestTarget,
                              NULL,
                              NULL);

   // clean up.
   pRenderTarget->Release();
   pDestTarget->Release();
}

Merci. J'ai entendu parler de cette méthode il y a quelque temps que l'on disait plus rapide que la lecture à partir du tampon frontal. Le faites-vous honnêtement de cette façon et cela fonctionne-t-il correctement?
someguy

Le problème avec le tampon frontal est un problème d'accès, c'est-à-dire d'essayer de copier un plain qui en cours de rendu "interrompt" la copie. Cela fonctionne assez bien pour moi et mange mon disque dur!
Brandrew

@bobobobo Je ne sais pas comment cela fonctionnerait exactement, mais je pensais utiliser quelque chose comme Huffyuv. Modifier: ou peut-être laisser l'utilisateur choisir parmi les filtres de directshow disponibles.
someguy

1
Je n'arrive tout simplement pas à faire ce travail ... DirectX crache un appel invalide sur la partie GetRenderTargetData. De toute évidence, la façon dont vous créez votre appareil doit avoir beaucoup d'importance.
LightStriker

12
downvote, cela ne fonctionne que pour votre propre application, donc ne peut pas être utilisé pour enregistrer des programmes génériques
user3125280

32

EDIT: Je peux voir que cela est répertorié sous votre premier lien d'édition comme "la manière GDI". C'est toujours une manière décente d'aller même avec l'avis de performance sur ce site, vous pouvez facilement atteindre 30fps, je pense.

D'après ce commentaire (je n'ai aucune expérience en la matière, je fais simplement référence à quelqu'un qui le fait):

HDC hdc = GetDC(NULL); // get the desktop device context
HDC hDest = CreateCompatibleDC(hdc); // create a device context to use yourself

// get the height and width of the screen
int height = GetSystemMetrics(SM_CYVIRTUALSCREEN);
int width = GetSystemMetrics(SM_CXVIRTUALSCREEN);

// create a bitmap
HBITMAP hbDesktop = CreateCompatibleBitmap( hdc, width, height);

// use the previously created device context with the bitmap
SelectObject(hDest, hbDesktop);

// copy from the desktop device context to the bitmap device context
// call this once per 'frame'
BitBlt(hDest, 0,0, width, height, hdc, 0, 0, SRCCOPY);

// after the recording is done, release the desktop context you got..
ReleaseDC(NULL, hdc);

// ..delete the bitmap you were using to capture frames..
DeleteObject(hbDesktop);

// ..and delete the context you created
DeleteDC(hDest);

Je ne dis pas que c'est le plus rapide, mais l' BitBltopération est généralement très rapide si vous copiez entre des contextes d'appareils compatibles.

Pour référence, Open Broadcaster Software implémente quelque chose comme ça dans le cadre de sa méthode "dc_capture" , bien qu'au lieu de créer le contexte de destination en hDestutilisant, CreateCompatibleDCils utilisent un IDXGISurface1, qui fonctionne avec DirectX 10+. S'il n'y a pas de soutien pour cela, ils se rabattent CreateCompatibleDC.

Pour le changer pour utiliser une application spécifique, vous devez changer la première ligne en GetDC(game)où se gametrouve la poignée de la fenêtre du jeu, puis définir la droite heightet widthla fenêtre du jeu également.

Une fois que vous avez les pixels dans hDest / hbDesktop, vous devez toujours les enregistrer dans un fichier, mais si vous effectuez une capture d'écran, je pense que vous voudriez en mettre un certain nombre en mémoire tampon et les enregistrer dans le fichier vidéo en morceaux, je ne pointerai donc pas vers le code pour enregistrer une image statique sur le disque.


3
msdn.microsoft.com/en-us/library/dd183370%28VS.85%29.aspx Extrait: si les formats de couleur des contextes de périphérique source et de destination ne correspondent pas, la fonction BitBlt convertit le format de couleur source pour correspondre à la destination format.

2
Des preuves pour étayer cela seraient bonnes. Avez-vous publié une comparaison des performances quelque part ou en avez-vous vu une exacte?

2
Essayez de le profiler. Comme je l'ai dit dans l'article, je fais référence à quelqu'un qui a de l'expérience avec GDI. S'il y a une fuite de mémoire et que vous savez comment le réparer, modifiez l'article pour éliminer la fuite.

1
J'ai obtenu environ 5 images par seconde avec cette méthode. C'était sur un ordinateur portable avec des graphiques intégrés, donc je m'attendrais à mieux d'un ordinateur de bureau avec une vraie carte graphique, mais quand même, c'est très lent.
Timmmm

1
@Timmmm J'ai ajouté une référence à la façon dont OBS implémente cela. Espérons que cela accélérera un peu les choses pour vous.

19

J'ai écrit un logiciel de capture vidéo, similaire à FRAPS pour les applications DirectX. Le code source est disponible et mon article explique la technique générale. Regardez http://blog.nektra.com/main/2013/07/23/instrumenting-direct3d-applications-to-capture-video-and-calculate-frames-per-second/

Respect de vos questions liées à la performance,

  • DirectX devrait être plus rapide que GDI, sauf lorsque vous lisez à partir du tampon frontal qui est très lent. Mon approche est similaire à FRAPS (lecture de backbuffer). J'intercepte un ensemble de méthodes des interfaces Direct3D.

  • Pour l'enregistrement vidéo en temps réel (avec un impact minimal sur l'application), un codec rapide est essentiel. FRAPS utilise son propre codec vidéo sans perte. Lagarith et HUFFYUV sont des codecs vidéo génériques sans perte conçus pour les applications en temps réel. Vous devriez les regarder si vous souhaitez sortir des fichiers vidéo.

  • Une autre approche pour enregistrer des screencasts pourrait être d'écrire un pilote miroir. Selon Wikipedia: Lorsque la mise en miroir vidéo est active, chaque fois que le système dessine vers le périphérique vidéo principal à un emplacement à l'intérieur de la zone en miroir, une copie de l'opération de dessin est exécutée sur le périphérique vidéo en miroir en temps réel. Consultez les pilotes miroir sur MSDN: http://msdn.microsoft.com/en-us/library/windows/hardware/ff568315(v=vs.85).aspx .


16

J'utilise d3d9 pour obtenir le backbuffer et je l'enregistre dans un fichier png en utilisant la bibliothèque d3dx:

    IDirect3DSurface9 * surface;

    // GetBackBuffer
    idirect3ddevice9-> GetBackBuffer (0, 0, D3DBACKBUFFER_TYPE_MONO, & surface);

    // enregistre la surface
    D3DXSaveSurfaceToFileA ("filename.png", D3DXIFF_PNG, surface, NULL, NULL);

    SAFE_RELEASE (surface);

Pour ce faire, vous devez créer votre swapbuffer avec

d3dpps.SwapEffect = D3DSWAPEFFECT_COPY ; // for screenshots.

(Vous garantissez donc que le backbuffer n'est pas mutilé avant de prendre la capture d'écran).


Cela a du sens, merci. Savez-vous quelle est la différence entre ceci et GetRenderTargetest?
someguy

Cela obtient juste la cible de rendu actuelle (pourrait être une autre surface hors écran si quelqu'un effectue le rendu en texture au moment où vous appelez).
bobobobo

13

Dans mon impression, l'approche GDI et l'approche DX sont de nature différente. la peinture à l'aide de GDI applique la méthode FLUSH, l'approche FLUSH dessine le cadre puis l'efface et redessine une autre image dans le même tampon, cela entraînera un scintillement dans les jeux nécessitant une fréquence d'images élevée.

  1. POURQUOI DX plus rapide? dans DX (ou monde graphique), une méthode plus mature appelée rendu à double tampon est appliquée, où deux tampons sont présents, lorsque le tampon avant est présent au matériel, vous pouvez également effectuer le rendu vers l'autre tampon, puis après l'image 1 fini le rendu, le système passe à l'autre tampon (le verrouille pour la présentation au matériel et libère le tampon précédent), de cette façon l'inefficacité du rendu est grandement améliorée.
  2. POURQUOI réduire l'accélération matérielle plus rapidement? bien qu'avec le rendu double tampon, le FPS est amélioré, mais le temps de rendu est encore limité. le matériel graphique moderne implique généralement beaucoup d'optimisation lors du rendu, généralement comme l'anti-aliasing, cela demande beaucoup de calcul, si vous n'avez pas besoin de graphiques de haute qualité, vous pouvez bien sûr simplement désactiver cette option. et cela vous fera gagner du temps.

Je pense que ce dont vous avez vraiment besoin, c'est d'un système de relecture, dont je suis totalement d'accord avec ce dont les gens ont discuté.


1
Voir la discussion sur les raisons pour lesquelles un système de relecture n'est pas faisable. Le programme de screencasting n'est pas destiné à un jeu spécifique.
someguy

10

J'ai écrit une classe qui implémentait la méthode GDI pour la capture d'écran. Moi aussi, je voulais une vitesse supplémentaire, donc après avoir découvert la méthode DirectX (via GetFrontBuffer), j'ai essayé cela, en m'attendant à ce qu'elle soit plus rapide.

J'ai été consterné de constater que GDI fonctionne environ 2,5 fois plus vite. Après 100 essais capturant mon affichage à deux moniteurs, l'implémentation GDI en moyenne 0,65 s par capture d'écran, tandis que la méthode DirectX en moyenne 1,72 s. Donc, GDI est définitivement plus rapide que GetFrontBuffer, d'après mes tests.

Je n'ai pas pu faire fonctionner le code de Brandrew pour tester DirectX via GetRenderTargetData. La copie d'écran est sortie purement noire. Cependant, il pourrait copier cet écran vierge très rapidement! Je continuerai à bricoler et j'espère obtenir une version fonctionnelle pour en voir les vrais résultats.


Merci pour l'information. Je n'ai pas testé le code de Brandrew, mais je sais que cette GetRenderTargetDataapproche fonctionne. J'écrirai peut-être ma propre réponse lorsque j'aurai terminé ma candidature. Ou, vous pouvez mettre à jour le vôtre une fois que tout fonctionne.
someguy

0,65 par capture d'écran?! Une bonne implémentation GDI (garder les appareils à portée de main, etc.) devrait faire 30fps en 1920x1200 facilement sur un ordinateur moderne.
Christopher Oezbek

Je suppose que la qualité d'image rendue par GDI est certainement plus mauvaise que DX
zinking

J'ai effectué ce test en C # avec SlimDX et, étonnamment, j'ai trouvé les mêmes résultats. Cela peut peut-être être dû au fait que, en utilisant SlimDX, il faut créer un nouveau flux et un nouveau bitmap pour chaque mise à jour d'image, au lieu de le créer une fois, de rembobiner et de continuer à écraser le même emplacement.
Cesar

Juste une clarification: en fait, les "mêmes résultats" faisaient référence à GDI étant plus rapide - comme @Christopher l'a mentionné, 30fps + était très faisable et laissait encore beaucoup de CPU de rechange.
Cesar

8

Quelques choses que j'ai pu glaner: apparemment, utiliser un "pilote miroir" est rapide bien que je ne connaisse pas de pilote OSS.

Pourquoi RDP est-il si rapide par rapport à d'autres logiciels de contrôle à distance?

Apparemment, l'utilisation de certaines convolutions de StretchRect est plus rapide que BitBlt

http://betterlogic.com/roger/2010/07/fast-screen-capture/comment-page-1/#comment-5193

Et celui que vous avez mentionné (les fraps se connectant aux dll D3D) est probablement le seul moyen pour les applications D3D, mais ne fonctionnera pas avec la capture de bureau Windows XP. Alors maintenant, je souhaite juste qu'il y ait un équivalent de fraps en termes de vitesse pour les fenêtres de bureau normales ... n'importe qui?

(Je pense qu'avec aero, vous pourrez peut-être utiliser des crochets de type fraps, mais les utilisateurs de XP n'auraient pas de chance).

Changer apparemment également la profondeur de bits de l'écran et / ou désactiver l'accélération matérielle. peut aider (et / ou désactiver l'aéro).

https://github.com/rdp/screen-capture-recorder-program comprend un utilitaire de capture basé sur BitBlt raisonnablement rapide et un benchmark dans le cadre de son installation, qui peut vous permettre de comparer les vitesses BitBlt pour les optimiser.

VirtualDub a également un module de capture d'écran "opengl" qui est dit rapide et fait des choses comme la détection des changements http://www.virtualdub.org/blog/pivot/entry.php?id=290


Je me demande, sera-t-il plus rapide d'utiliser votre "enregistreur de capture d'écran" ou d'utiliser BitBlt moi-même? Y a-t-il une optimisation dans votre projet?
blez

Si vous utilisez Bitblt vous-même, vous pourriez éviter un memcpy supplémentaire (memcpy n'est généralement pas le plus gros goulot d'étranglement, bien que cela ajoute du temps - je ne le mentionnais ici que pour son utilitaire de référence, mais si vous avez besoin de quelque chose ou de directshow, c'est bien )
rogerdpack

8

Pour C ++, vous pouvez utiliser: http://www.pinvoke.net/default.aspx/gdi32/BitBlt.html
Cela peut ne pas fonctionner sur tous les types d'applications 3D / applications vidéo. Ensuite, ce lien peut être plus utile car il décrit 3 méthodes différentes que vous pouvez utiliser.

Ancienne réponse (C #):
Vous pouvez utiliser System.Drawing.Graphics.Copy , mais ce n'est pas très rapide.

Un exemple de projet que j'ai écrit en faisant exactement ceci: http://blog.tedd.no/index.php/2010/08/16/c-image-analysis-auto-gaming-with-source/

Je prévois de mettre à jour cet exemple en utilisant une méthode plus rapide comme Direct3D: http://spazzarama.com/2009/02/07/screencapture-with-direct3d/

Et voici un lien pour capturer en vidéo: Comment capturer l'écran en vidéo en utilisant C # .Net?


2
Ah, j'ai oublié de mentionner que je programme en C (éventuellement C ++) et que je ne prévois pas d'utiliser .NET. Terriblement désolé :/.
someguy

J'étais déjà au courant de BitBlt (GDI). Je vais cependant examiner Direct3D. Merci!
someguy

J'examinais cela il y a quelques semaines, mais je n'ai pas encore réussi à l'implémenter. Direct3D est !! façon !! plus rapide que la méthode intégrée C # qui utilise GDI +.
Tedd Hansen

J'ai mis à jour mon message d'origine. Selon le lien, DirectX est lent lorsqu'il faut appeler GetFrontBufferData (). Est-ce quelque chose à considérer lors de l'enregistrement des séquences de jeu? Pourriez-vous contextualiser cela pour moi?
someguy

1
GDI est lent, donc il ne convient pas au domaine du problème, DirectX ou OpenGL serait la seule recommandation raisonnable.

6

Avec Windows 8, Microsoft a introduit l'API de duplication du bureau Windows. C'est la manière officiellement recommandée de le faire. Une fonctionnalité intéressante pour la capture d'écran est qu'il détecte le mouvement de la fenêtre, vous pouvez donc transmettre des deltas de bloc lorsque les fenêtres sont déplacées, au lieu de pixels bruts. En outre, il vous indique quels rectangles ont changé, d'une image à l'autre.

L'exemple de code Microsoft est assez complexe, mais l'API est en fait simple et facile à utiliser. J'ai mis en place un exemple de projet qui est beaucoup plus simple que l'exemple officiel:

https://github.com/bmharper/WindowsDesktopDuplicationSample

Docs: https://docs.microsoft.com/en-gb/windows/desktop/direct3ddxgi/desktop-dup-api

Exemple de code officiel Microsoft: https://code.msdn.microsoft.com/windowsdesktop/Desktop-Duplication-Sample-da4c696a


Le projet Github fonctionne parfaitement sur Windows 10, testé avec la vidéo Web et Resident Evil 7. La meilleure chose est que la charge du processeur évolue avec le taux de mise à niveau graphique.
jw_

dès que vous lancez GPU-Z, ce programme cesse de saisir l'écran.
Marino Šimić

5

Vous pouvez essayer le projet open source c ++ WinRobot @git , un puissant capteur d' écran

CComPtr<IWinRobotService> pService;
hr = pService.CoCreateInstance(__uuidof(ServiceHost) );

//get active console session
CComPtr<IUnknown> pUnk;
hr = pService->GetActiveConsoleSession(&pUnk);
CComQIPtr<IWinRobotSession> pSession = pUnk;

// capture screen
pUnk = 0;
hr = pSession->CreateScreenCapture(0,0,1280,800,&pUnk);

// get screen image data(with file mapping)
CComQIPtr<IScreenBufferStream> pBuffer = pUnk;

Soutien :

  • Fenêtre UAC
  • Winlogon
  • DirectShowOverlay

1
C'est ... beaucoup d'hameçons de bas niveau. La vitesse de capture est incroyable si vous avez des droits d'administrateur.
toster-cx

J'ai trouvé le winrobot très puissant et fluide.
ashishgupta_mca

J'ai étudié le code WinRobot et je n'ai rien vu de capture d'écran révolutionnaire: il utilise le même CreateCompatibleDC..BitBlt. À moins qu'il n'y ait de la magie lorsque cela est effectué dans un contexte de service?
shekh

@shekh: Votre étude était trop superficielle. Le code utilise IDirectDrawSurface7-> BltFast () pour copier l'écran de la surface de dessin d'écran vers une surface DD de copie, puis il utilise un mappage de fichiers pour copier l'image. C'est assez complexe car le code s'exécute dans un service où vous ne pouvez pas facilement accéder aux bureaux.
Elmue

2

Je le fais moi-même avec directx et je pense que c'est aussi rapide que vous le souhaiteriez. Je n'ai pas d'exemple de code rapide, mais j'ai trouvé cela qui devrait être utile. la version directx11 ne devrait pas différer beaucoup, directx9 peut-être un peu plus, mais c'est la voie à suivre


2

Je me rends compte que la suggestion suivante ne répond pas à votre question, mais la méthode la plus simple que j'ai trouvée pour capturer une vue DirectX en évolution rapide est de brancher une caméra vidéo sur le port S-vidéo de la carte vidéo et d'enregistrer les images comme un film. Transférez ensuite la vidéo de l'appareil photo vers un fichier MPG, WMV, AVI, etc. sur l'ordinateur.


2

L'enregistrement d'écran peut être effectué en C # à l' aide de l' API VLC . J'ai fait un exemple de programme pour le démontrer. Il utilise LibVLCSharp et VideoLAN.LibVLC.Windows bibliothèques . Vous pouvez obtenir de nombreuses autres fonctionnalités liées au rendu vidéo à l'aide de cette API multiplateforme.

Pour la documentation de l'API, voir: API LibVLCSharp Github

using System;
using System.IO;
using System.Reflection;
using System.Threading;
using LibVLCSharp.Shared;

namespace ScreenRecorderNetApp
{
    class Program
    {
        static void Main(string[] args)
        {
            Core.Initialize();

            using (var libVlc = new LibVLC())
            using (var mediaPlayer = new MediaPlayer(libVlc))
            {
                var media = new Media(libVlc, "screen://", FromType.FromLocation);
                media.AddOption(":screen-fps=24");
                media.AddOption(":sout=#transcode{vcodec=h264,vb=0,scale=0,acodec=mp4a,ab=128,channels=2,samplerate=44100}:file{dst=testvlc.mp4}");
                media.AddOption(":sout-keep");

                mediaPlayer.Play(media);
                Thread.Sleep(10*1000);
                mediaPlayer.Stop();
            }
        }
    }
}

1
À qui cela peut concerner: Veuillez noter que libvlc est publié sous GPL. Si vous n'avez pas l'intention de publier votre code sous GPL, n'utilisez pas libvlc.
Sebastian Cabot

Ce n'est pas tout à fait exact, LibVLC est LGPL - il a été renouvelé en 2011. L'application VLC elle-même reste GPL: videolan.org/press/lgpl-libvlc.html
Andrew le

0

Capture de bureau DXGI

Projet qui capture l'image du bureau avec duplication DXGI. Enregistre l'image capturée dans le fichier dans différents formats d'image (* .bmp; * .jpg; * .tif).

Cet exemple est écrit en C ++. Vous avez également besoin d'une certaine expérience avec DirectX (D3D11, D2D1).

Ce que l'application peut faire

  • Si vous avez plusieurs moniteurs de bureau, vous pouvez choisir.
  • Redimensionnez l'image de bureau capturée.
  • Choisissez différents modes de mise à l'échelle.
  • Vous pouvez afficher ou masquer l'icône de la souris dans l'image de sortie.
  • Vous pouvez faire pivoter l'image de l'image de sortie ou la laisser par défaut.
En utilisant notre site, vous reconnaissez avoir lu et compris notre politique liée aux cookies et notre politique de confidentialité.
Licensed under cc by-sa 3.0 with attribution required.