mirror of
https://github.com/alfishe/amiga-bootcamp.git
synced 2026-06-12 16:16:28 +00:00
992 lines
38 KiB
Markdown
992 lines
38 KiB
Markdown
[← Home](../README.md) · [Intuition](README.md)
|
||
|
||
# Screens
|
||
|
||
## Overview
|
||
|
||
An Amiga **Screen** is a complete, independent display context. Each screen owns its own `ViewPort`, `BitMap`, color palette, resolution, and refresh rate. Unlike modern desktops where all applications share a single framebuffer, the Amiga allows **multiple screens with different resolutions and color depths to coexist simultaneously** — and the user can drag them up and down to reveal each other in real time.
|
||
|
||
This is one of the defining features of the Amiga. No other mainstream personal computer of the 1985–1995 era offered anything comparable.
|
||
|
||
---
|
||
|
||
## How Screens Work — The Copper Connection
|
||
|
||
The hardware trick that makes Amiga screens possible is the **[Copper](../08_graphics/copper/copper.md)** (coprocessor), a DMA-driven programmable sequencer inside the Agnus/Alice chip. The Copper executes a list of instructions synchronized to the CRT electron beam position (see also: [Copper Programming](../08_graphics/copper/copper_programming.md), [OCS Copper](../01_hardware/ocs_a500/copper.md), [AGA Copper](../01_hardware/aga_a1200_a4000/aga_copper.md)):
|
||
|
||
| Instruction | Action |
|
||
|---|---|
|
||
| **WAIT** | Pause until the beam reaches a specific `(x, y)` position |
|
||
| **MOVE** | Write a value to a custom chip register (palette, bitplane pointers, resolution) |
|
||
| **SKIP** | Conditional branch based on beam position |
|
||
|
||
When multiple screens are open, Intuition builds a **Copper list** that changes display parameters at the scanline boundary between screens. The screen on top uses one set of registers (resolution, palette, bitplane pointers), and when the beam crosses into the next screen's area, the Copper switches registers mid-frame. This is why:
|
||
|
||
- A 640×256 4-color Workbench screen can sit above a 320×256 32-color game screen
|
||
- Dragging a screen changes the Copper's `WAIT` coordinates — the boundary moves in real time
|
||
- There is **zero CPU cost** for maintaining multiple screens — the Copper handles it via DMA
|
||
|
||
```mermaid
|
||
sequenceDiagram
|
||
participant Beam as CRT Beam
|
||
participant Copper as Copper Coprocessor
|
||
participant Regs as Custom Chip Registers
|
||
|
||
Note over Beam,Regs: Frame begins
|
||
Copper->>Copper: WAIT y=0
|
||
Copper->>Regs: MOVE BPLCON0 = HiRes, 4 colors
|
||
Copper->>Regs: MOVE COLOR00-03 = Workbench palette
|
||
Copper->>Regs: MOVE BPLxPT = Workbench bitplanes
|
||
|
||
Note over Beam,Regs: Beam reaches screen boundary
|
||
Copper->>Copper: WAIT y=128
|
||
Copper->>Regs: MOVE BPLCON0 = LoRes, 32 colors
|
||
Copper->>Regs: MOVE COLOR00-31 = Game palette
|
||
Copper->>Regs: MOVE BPLxPT = Game bitplanes
|
||
|
||
Note over Beam,Regs: Frame ends, VBlank
|
||
```
|
||
|
||
---
|
||
|
||
## Screen Types
|
||
|
||
| Type | Constant | Description |
|
||
|---|---|---|
|
||
| **Workbench** | `WBENCHSCREEN` | The default desktop screen — always exists while Workbench is running |
|
||
| **Custom** | `CUSTOMSCREEN` | Application-owned, private screen — only the creating program uses it |
|
||
| **Public** | `PUBLICSCREEN` | Named screen that any application can open windows on |
|
||
|
||
### Workbench Screen
|
||
|
||
The Workbench screen is the system's home screen. It defaults to the user's configured resolution (typically HiRes 640×256 PAL or 640×200 NTSC with 4 colors). Users can change the Workbench screen mode through the **ScreenMode** Preferences editor.
|
||
|
||
### Custom Screens
|
||
|
||
Applications that need specific display characteristics — games, paint programs, video editors — open their own custom screen. The application controls the resolution, depth, palette, and lifetime. Only that application's windows appear on it.
|
||
|
||
### Public Screens
|
||
|
||
Introduced in AmigaOS 2.0, public screens solve the memory waste of every application opening its own screen. An application can:
|
||
|
||
1. Open a screen and register it with a name (`SA_PubName`)
|
||
2. Other applications can discover and open windows on it via `LockPubScreen()`
|
||
3. The Workbench screen itself is the default public screen
|
||
|
||
---
|
||
|
||
## Display Modes and Resolutions
|
||
|
||
### Chipset Generations
|
||
|
||
| Chipset | Era | Machines | Key Display Features |
|
||
|---|---|---|---|
|
||
| **OCS** | 1985 | A1000, A500, A2000 | LoRes/HiRes, 32 colors from 4,096 (12-bit), HAM, EHB |
|
||
| **ECS** | 1990 | A500+, A600, A3000 | + SuperHires, Productivity mode (31kHz VGA), improved genlock |
|
||
| **AGA** | 1992 | A1200, A4000, CD32 | 256 colors from 16.7M (24-bit), HAM-8, mode promotion |
|
||
| **RTG** | 1994+ | Zorro/PCI cards | True-color (16/24/32-bit), arbitrary resolutions via CyberGraphX / Picasso96 |
|
||
|
||
### Native Resolution Table
|
||
|
||
| Mode | NTSC (60 Hz) | PAL (50 Hz) | Bitplanes | Colors |
|
||
|---|---|---|---|---|
|
||
| **LoRes** | 320 × 200 | 320 × 256 | 1–5 | 2–32 |
|
||
| **LoRes Interlaced** | 320 × 400 | 320 × 512 | 1–5 | 2–32 |
|
||
| **HiRes** | 640 × 200 | 640 × 256 | 1–4 | 2–16 |
|
||
| **HiRes Interlaced** | 640 × 400 | 640 × 512 | 1–4 | 2–16 |
|
||
| **SuperHires** (ECS+) | 1280 × 200 | 1280 × 256 | 1–2 | 2–4 |
|
||
| **SuperHires Interlaced** (ECS+) | 1280 × 400 | 1280 × 512 | 1–2 | 2–4 |
|
||
| **Productivity** (ECS+) | 640 × 480 | 640 × 480 | 1–4 | 2–16 |
|
||
| **AGA LoRes** | 320 × 200 | 320 × 256 | 1–8 | 2–256 |
|
||
| **AGA HiRes** | 640 × 200 | 640 × 256 | 1–8 | 2–256 |
|
||
|
||
### Special Modes
|
||
|
||
| Mode | Description |
|
||
|---|---|
|
||
| **HAM** (Hold-And-Modify) | OCS/ECS: displays all 4,096 colors. Each pixel modifies one RGB component of its neighbor |
|
||
| **HAM-8** | AGA: displays ~262,000 colors simultaneously using 8 bitplanes |
|
||
| **EHB** (Extra Half-Brite) | 64 colors — 32 programmer-defined + 32 at half brightness |
|
||
| **DBLPAL / DBLNTSC** | AGA doubled scan rates for flicker-free interlaced-resolution output |
|
||
|
||
### Display IDs
|
||
|
||
The graphics library maintains a **display database**. Each mode is identified by a 32-bit **Display ID** (also called ModeID):
|
||
|
||
```c
|
||
/* Common Display IDs from <graphics/displayinfo.h> */
|
||
#define LORES_KEY 0x00000000
|
||
#define HIRES_KEY 0x00008000
|
||
#define SUPER_KEY 0x00008020
|
||
#define HAM_KEY 0x00000800
|
||
#define LORESLACE_KEY 0x00000004
|
||
#define HIRESLACE_KEY 0x00008004
|
||
#define SUPERLACE_KEY 0x00008024
|
||
#define LORESDPF_KEY 0x00000400 /* Dual Playfield */
|
||
|
||
/* PAL variants add PAL_MONITOR_ID (0x00021000) */
|
||
#define PAL_MONITOR_ID 0x00021000
|
||
```
|
||
|
||
Applications query available modes with `NextDisplayInfo()` and `GetDisplayInfoData()`. The ScreenMode preferences editor lets users browse all available modes.
|
||
|
||
### RTG — Breaking Free of Native Hardware
|
||
|
||
With graphics cards (Picasso II, CyberVision 64, etc.), RTG libraries (CyberGraphX, Picasso96) register additional Display IDs. These enable:
|
||
|
||
- **Arbitrary resolutions**: 800×600, 1024×768, 1280×1024, etc.
|
||
- **True color**: 15-bit, 16-bit, 24-bit, 32-bit
|
||
- **No Copper involvement** — the graphics card has its own framebuffer
|
||
- Applications using standard `OpenScreenTags()` with an RTG DisplayID work transparently
|
||
|
||
```c
|
||
/* Open an 800x600 16-bit screen on a graphics card */
|
||
struct Screen *scr = OpenScreenTags(NULL,
|
||
SA_Width, 800,
|
||
SA_Height, 600,
|
||
SA_Depth, 16,
|
||
SA_DisplayID, 0x50041000, /* Picasso96 mode ID */
|
||
SA_Title, "RTG Screen",
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Pens, (ULONG)~0,
|
||
TAG_DONE);
|
||
```
|
||
|
||
---
|
||
|
||
## Screen Dragging — User Interaction
|
||
|
||
### How It Works
|
||
|
||
Every screen has a **drag bar** (the title bar area). The user clicks and drags it to slide the screen vertically:
|
||
|
||
- **Drag down** — reveals the screen behind it
|
||
- **Drag up** — hides the screen behind it
|
||
- **Left-Amiga + mouse drag** — drags even if the title bar is obscured by windows
|
||
|
||
When the user drags, Intuition updates the Copper list's `WAIT` coordinates in real time. The "seam" between two screens moves smoothly without any redrawing — the bitplanes never change, only the vertical offset in the display list changes.
|
||
|
||
### Screen Depth Arrangement
|
||
|
||
| Action | Keyboard Shortcut | Effect |
|
||
|---|---|---|
|
||
| Screen to front | Left-Amiga + M (or depth gadget) | Moves screen to top of stack |
|
||
| Screen to back | Left-Amiga + N (or depth gadget) | Moves screen to bottom of stack |
|
||
| Drag screen | Left-Amiga + mouse drag | Slides screen vertically |
|
||
|
||
### Real-World Usage Patterns
|
||
|
||
**Pattern 1: Workbench + Game**
|
||
|
||
The most common Amiga experience. The user launches a game that opens a 320×256 32-color custom screen. The Workbench (640×256 4-color) sits behind it. The user can drag the game screen down to check the Workbench clock, read a README in a shell, or access a file manager — while the game continues running.
|
||
|
||
**Pattern 2: Productivity Multi-Screen**
|
||
|
||
A developer in 1994 might have:
|
||
- Screen 1: **CygnusEd** (text editor) on HiRes 640×512 interlaced, 8 colors
|
||
- Screen 2: **Deluxe Paint** on LoRes 320×256, 32 colors
|
||
- Screen 3: **Workbench** with shell windows, file manager
|
||
|
||
Each screen has exactly the resolution and color depth it needs. Dragging between them is instant.
|
||
|
||
**Pattern 3: Video Production**
|
||
|
||
Amiga video professionals (Babylon 5, MTV, Weather Channel) routinely used:
|
||
- A **genlock** screen for video overlay compositing
|
||
- A **HAM** screen in Deluxe Paint for photorealistic image editing
|
||
- The Workbench screen for file management
|
||
|
||
All three coexist and are draggable.
|
||
|
||
**Pattern 4: Public Screen Sharing**
|
||
|
||
An email client (YAM) opens a MUI public screen. The user configures their address book, text editor, and web browser (IBrowse) to all open windows on the same public screen — sharing a single set of bitplanes instead of each application consuming ~300 KB for its own screen.
|
||
|
||
---
|
||
|
||
## Opening a Screen (OS 2.0+ TagList)
|
||
|
||
```c
|
||
struct Screen *scr = OpenScreenTags(NULL,
|
||
SA_Width, 640,
|
||
SA_Height, 256,
|
||
SA_Depth, 4, /* 16 colors */
|
||
SA_DisplayID, HIRES_KEY,
|
||
SA_Title, "My Screen",
|
||
SA_ShowTitle, TRUE,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Pens, (ULONG)~0, /* use 3D pen defaults */
|
||
SA_Font, &topaz8,
|
||
TAG_DONE);
|
||
```
|
||
|
||
### Common SA_ Tags
|
||
|
||
| Tag | Type | Description |
|
||
|---|---|---|
|
||
| `SA_Width`, `SA_Height` | `UWORD` | Screen dimensions in pixels |
|
||
| `SA_Depth` | `UWORD` | Bitplane count (1–8 native, 15/16/24 RTG) |
|
||
| `SA_DisplayID` | `ULONG` | ModeID from display database |
|
||
| `SA_Title` | `STRPTR` | Title bar text |
|
||
| `SA_ShowTitle` | `BOOL` | Show/hide title bar |
|
||
| `SA_Type` | `UWORD` | `CUSTOMSCREEN`, `PUBLICSCREEN` |
|
||
| `SA_Behind` | `BOOL` | Open behind other screens |
|
||
| `SA_Quiet` | `BOOL` | No title bar at all |
|
||
| `SA_AutoScroll` | `BOOL` | Allow autoscrolling when screen > display |
|
||
| `SA_Overscan` | `ULONG` | `OSCAN_TEXT`, `OSCAN_STANDARD`, `OSCAN_MAX`, `OSCAN_VIDEO` |
|
||
| `SA_Pens` | `UWORD *` | DrawInfo pen array for 3D look (terminate with `~0`) |
|
||
| `SA_Interleaved` | `BOOL` | Use interleaved bitmap (better blitter performance) |
|
||
| `SA_Colors32` | `ULONG *` | 32-bit RGB color palette (AGA/RTG) |
|
||
| `SA_PubName` | `STRPTR` | Public screen name for `LockPubScreen()` |
|
||
| `SA_Font` | `struct TextAttr *` | Screen default font |
|
||
| `SA_SysFont` | `UWORD` | Use system default font (0 = fixed, 1 = proportional) |
|
||
| `SA_MinimizeISG` | `BOOL` | Minimize inter-screen gap during dragging |
|
||
|
||
### Overscan Modes
|
||
|
||
| Constant | Description |
|
||
|---|---|
|
||
| `OSCAN_TEXT` | Maximum area that can display text clearly |
|
||
| `OSCAN_STANDARD` | Default overscan — fits most monitors |
|
||
| `OSCAN_MAX` | Maximum hardware overscan — may clip on some monitors |
|
||
| `OSCAN_VIDEO` | Full video overscan for broadcast/genlock use |
|
||
|
||
---
|
||
|
||
## Closing a Screen
|
||
|
||
```c
|
||
/* All windows on this screen must be closed first! */
|
||
if (CloseScreen(scr))
|
||
{
|
||
/* Screen closed successfully */
|
||
}
|
||
else
|
||
{
|
||
/* Failed — a window is still open on this screen,
|
||
or another application has locked it */
|
||
}
|
||
```
|
||
|
||
For public screens, you must also call `PubScreenStatus(scr, PSNF_PRIVATE)` before closing to prevent new windows from opening on it.
|
||
|
||
---
|
||
|
||
## Public Screens
|
||
|
||
```c
|
||
/* === Creating a public screen === */
|
||
struct Screen *pub = OpenScreenTags(NULL,
|
||
SA_Type, PUBLICSCREEN,
|
||
SA_PubName, "MYAPP",
|
||
SA_Title, "My Application Screen",
|
||
SA_Depth, 4,
|
||
SA_DisplayID, HIRES_KEY,
|
||
SA_Pens, (ULONG)~0,
|
||
TAG_DONE);
|
||
|
||
/* Make it visible to other applications */
|
||
PubScreenStatus(pub, 0);
|
||
|
||
/* === Opening a window on someone else's public screen === */
|
||
struct Screen *found = LockPubScreen("MYAPP");
|
||
if (found)
|
||
{
|
||
struct Window *win = OpenWindowTags(NULL,
|
||
WA_PubScreen, found,
|
||
WA_Title, "Guest Window on MYAPP",
|
||
WA_InnerWidth, 300,
|
||
WA_InnerHeight, 200,
|
||
TAG_DONE);
|
||
|
||
UnlockPubScreen(NULL, found);
|
||
|
||
/* ... use window ... */
|
||
|
||
CloseWindow(win);
|
||
}
|
||
```
|
||
|
||
### Default Public Screen
|
||
|
||
If `LockPubScreen(NULL)` is called with a `NULL` name, it returns the **default public screen** — normally the Workbench screen. Applications that don't need their own screen should open windows on the default public screen:
|
||
|
||
```c
|
||
struct Screen *wb = LockPubScreen(NULL); /* Get Workbench */
|
||
/* open windows on wb */
|
||
UnlockPubScreen(NULL, wb);
|
||
```
|
||
|
||
---
|
||
|
||
## Screen Data Structures
|
||
|
||
### Key Fields of `struct Screen`
|
||
|
||
| Field | Type | Description |
|
||
|---|---|---|
|
||
| `LeftEdge`, `TopEdge` | `WORD` | Screen position (TopEdge changes during dragging) |
|
||
| `Width`, `Height` | `WORD` | Screen dimensions |
|
||
| `RastPort` | `struct RastPort` | Drawing context for the screen |
|
||
| `BitMap` | `struct BitMap *` | The screen's bitmap (bitplanes) |
|
||
| `ViewPort` | `struct ViewPort` | Display configuration (mode, colors) |
|
||
| `Title` | `UBYTE *` | Title bar string |
|
||
| `Font` | `struct TextFont *` | Default font |
|
||
| `FirstWindow` | `struct Window *` | Linked list of windows on this screen |
|
||
|
||
### Querying Screen Properties
|
||
|
||
```c
|
||
/* Get DrawInfo for 3D pen colors */
|
||
struct DrawInfo *dri = GetScreenDrawInfo(scr);
|
||
ULONG textpen = dri->dri_Pens[TEXTPEN];
|
||
ULONG shinepen = dri->dri_Pens[SHINEPEN];
|
||
ULONG shadowpen = dri->dri_Pens[SHADOWPEN];
|
||
/* ... */
|
||
FreeScreenDrawInfo(scr, dri);
|
||
```
|
||
|
||
---
|
||
|
||
## Input Event Routing
|
||
|
||
Understanding how mouse and keyboard events flow through the system is critical when working with multiple screens.
|
||
|
||
### The Input Pipeline
|
||
|
||
```mermaid
|
||
graph TB
|
||
HW["Hardware<br/>(mouse, keyboard)"] --> ID["input.device<br/>Creates InputEvent stream"]
|
||
ID --> CHAIN["Handler Chain<br/>(prioritized list)"]
|
||
CHAIN --> COMM["Commodities<br/>(hotkeys, priority 51)"]
|
||
COMM --> INT["Intuition<br/>(priority 50)"]
|
||
INT -->|"System action<br/>(screen drag, gadget)"| CONSUME["Consumed — app never sees it"]
|
||
INT -->|"App event"| ROUTE["Route to target window"]
|
||
ROUTE --> IDCMP["IDCMP Port<br/>(IntuiMessage)"]
|
||
IDCMP --> APP["Application<br/>GetMsg() / WaitPort()"]
|
||
|
||
style HW fill:#f5f5f5,stroke:#bdbdbd,color:#333
|
||
style INT fill:#e8f4fd,stroke:#2196f3,color:#333
|
||
style CONSUME fill:#ffebee,stroke:#f44336,color:#333
|
||
style APP fill:#e8f5e9,stroke:#4caf50,color:#333
|
||
```
|
||
|
||
### How It Works
|
||
|
||
1. **`input.device`** gathers raw hardware events (mouse position, button state, key scancodes) and packages them as `InputEvent` structures
|
||
2. Events pass through a **prioritized handler chain** — each handler can consume, modify, or pass events
|
||
3. **Intuition** sits at priority 50. It intercepts system-level actions:
|
||
- Screen dragging (Left-Amiga + mouse)
|
||
- Window depth/sizing/closing gadgets
|
||
- Menu activation (right mouse button)
|
||
4. Events that Intuition doesn't consume are routed to the **active window** based on mouse position
|
||
5. The window only receives events matching its **IDCMP flags** — everything else is dropped
|
||
|
||
### IDCMP — Intuition Direct Communication Message Port
|
||
|
||
Each window specifies which events it wants via IDCMP flags at creation time:
|
||
|
||
```c
|
||
struct Window *win = OpenWindowTags(NULL,
|
||
WA_IDCMP, IDCMP_MOUSEBUTTONS | IDCMP_RAWKEY |
|
||
IDCMP_GADGETUP | IDCMP_CLOSEWINDOW |
|
||
IDCMP_CHANGEWINDOW,
|
||
/* ... */
|
||
TAG_DONE);
|
||
```
|
||
|
||
| IDCMP Flag | Event |
|
||
|---|---|
|
||
| `IDCMP_MOUSEBUTTONS` | Mouse button press/release |
|
||
| `IDCMP_MOUSEMOVE` | Mouse position changes (high bandwidth — use sparingly) |
|
||
| `IDCMP_RAWKEY` | Raw keyboard scancodes |
|
||
| `IDCMP_VANILLAKEY` | Translated ASCII characters |
|
||
| `IDCMP_GADGETUP/DOWN` | Gadget interaction |
|
||
| `IDCMP_CLOSEWINDOW` | User clicked close gadget |
|
||
| `IDCMP_CHANGEWINDOW` | Window moved, resized, or screen changed |
|
||
| `IDCMP_ACTIVEWINDOW` | This window became the active window |
|
||
| `IDCMP_INACTIVEWINDOW` | This window lost focus |
|
||
| `IDCMP_SCREENPOSITION` | Screen was dragged (OS 3.0+) |
|
||
|
||
### Events During Screen Dragging
|
||
|
||
When the user drags a screen (Left-Amiga + mouse select), **Intuition consumes the input entirely**:
|
||
|
||
- The application does **not** receive `IDCMP_MOUSEBUTTONS` for the select-down
|
||
- The application does **not** receive `IDCMP_MOUSEMOVE` during the drag
|
||
- If the window has `IDCMP_CHANGEWINDOW` set, it receives a notification **after** the drag completes
|
||
- `IDCMP_SCREENPOSITION` (OS 3.0+) notifies windows that their screen moved — useful for applications that cache screen coordinates
|
||
|
||
### Keyboard Shortcuts and Screen Control
|
||
|
||
Intuition intercepts certain key combinations before applications see them:
|
||
|
||
| Shortcut | Action | Consumed by |
|
||
|---|---|---|
|
||
| Left-Amiga + M | Screen to front | Intuition |
|
||
| Left-Amiga + N | Screen to back | Intuition |
|
||
| Left-Amiga + mouse drag | Drag screen vertically | Intuition |
|
||
| Left-Amiga + left/right | Cycle between screens | Intuition (OS 3.0+) |
|
||
| Right-Amiga + keys | Application shortcuts | Application (via `IDCMP_RAWKEY`) |
|
||
|
||
> **Key insight**: Left-Amiga combinations are reserved for system use. Applications should use Right-Amiga for their own shortcuts to avoid conflicts.
|
||
|
||
### Event Routing Across Screens
|
||
|
||
When multiple screens are visible, mouse events are routed to the **frontmost screen under the pointer**:
|
||
|
||
- If the pointer is in the visible area of Screen A, events go to Screen A's active window
|
||
- If the user drags Screen A down and the pointer enters Screen B's visible area, events switch to Screen B
|
||
- Keyboard events always go to the **active window on the active screen**, regardless of pointer position
|
||
|
||
---
|
||
|
||
## Pitfalls and Hazards
|
||
|
||
### 1. CloseScreen Deadlock
|
||
|
||
`CloseScreen()` will **fail silently** if any window is still open on the screen:
|
||
|
||
```c
|
||
/* WRONG — will fail if visitor windows exist */
|
||
CloseScreen(myScreen);
|
||
|
||
/* RIGHT — check for visitor windows first */
|
||
if (!CloseScreen(myScreen))
|
||
{
|
||
/* Screen still has open windows!
|
||
Either wait, or set PSNF_PRIVATE to block new visitors */
|
||
PubScreenStatus(myScreen, PSNF_PRIVATE);
|
||
/* Try again after all visitors close */
|
||
}
|
||
```
|
||
|
||
For public screens, always transition to `PSNF_PRIVATE` before attempting to close.
|
||
|
||
### 2. IDCMP Message Leak
|
||
|
||
Every `IntuiMessage` received via `GetMsg()` **must** be replied with `ReplyMsg()`. Failure to reply causes:
|
||
|
||
- Intuition's message pool to fill up
|
||
- Eventually, the system hangs — Intuition blocks waiting for message slots
|
||
|
||
```c
|
||
/* WRONG — message leak */
|
||
struct IntuiMessage *msg = (struct IntuiMessage *)GetMsg(win->UserPort);
|
||
ULONG class = msg->Class;
|
||
/* forgot ReplyMsg! */
|
||
|
||
/* RIGHT — always reply */
|
||
struct IntuiMessage *msg;
|
||
while (msg = (struct IntuiMessage *)GetMsg(win->UserPort))
|
||
{
|
||
ULONG class = msg->Class;
|
||
UWORD code = msg->Code;
|
||
ReplyMsg((struct Message *)msg); /* Reply BEFORE processing */
|
||
|
||
switch(class) { /* ... */ }
|
||
}
|
||
```
|
||
|
||
### 3. Layer Lock Deadlock
|
||
|
||
Never call Intuition functions while holding a layer lock:
|
||
|
||
```c
|
||
/* DEADLOCK — Intuition needs the layer lock internally */
|
||
LockLayerInfo(&scr->LayerInfo);
|
||
OpenWindow(...); /* Intuition tries to lock the same LayerInfo → deadlock */
|
||
```
|
||
|
||
### 4. Screen Pointer Invalidation
|
||
|
||
Screen pointers can become invalid during iconification or if another process closes the screen:
|
||
|
||
```c
|
||
/* HAZARD — cached screen pointer may be stale */
|
||
struct Screen *scr = LockPubScreen("MYAPP");
|
||
UnlockPubScreen(NULL, scr);
|
||
/* ... time passes ... */
|
||
/* scr may no longer be valid! */
|
||
|
||
/* SAFE — re-lock when needed */
|
||
struct Screen *scr = LockPubScreen("MYAPP");
|
||
if (scr)
|
||
{
|
||
/* use scr */
|
||
UnlockPubScreen(NULL, scr);
|
||
}
|
||
```
|
||
|
||
### 5. Interlaced Flicker
|
||
|
||
Interlaced modes (400/512 vertical lines) flicker badly on CRT monitors when adjacent lines have high contrast. Mitigation:
|
||
|
||
- Use a **flicker fixer** (hardware scan doubler) like the A3000's built-in deinterlacer
|
||
- Avoid pure black/white adjacent lines in interlaced UIs
|
||
- **AGA mode promotion** can convert interlaced modes to flicker-free doubled-scan output on VGA monitors
|
||
|
||
### 6. MOUSEMOVE Flooding
|
||
|
||
Requesting `IDCMP_MOUSEMOVE` generates an event for **every pixel** the mouse moves. This can overwhelm slow applications:
|
||
|
||
```c
|
||
/* Use IDCMP_INTUITICKS instead for periodic polling */
|
||
WA_IDCMP, IDCMP_INTUITICKS, /* ~10 Hz timer, much lighter */
|
||
|
||
/* Or use WFLG_REPORTMOUSE only on specific gadgets */
|
||
```
|
||
|
||
### 7. Hardcoded Display IDs
|
||
|
||
Never hardcode Display IDs for screen modes. The user's hardware determines available modes:
|
||
|
||
```c
|
||
/* WRONG — assumes PAL HiRes exists */
|
||
SA_DisplayID, 0x00029000,
|
||
|
||
/* RIGHT — query the display database */
|
||
ULONG modeID = BestModeID(
|
||
BIDTAG_NominalWidth, 640,
|
||
BIDTAG_NominalHeight, 256,
|
||
BIDTAG_Depth, 4,
|
||
TAG_DONE);
|
||
|
||
if (modeID != INVALID_ID)
|
||
SA_DisplayID, modeID,
|
||
```
|
||
|
||
---
|
||
|
||
## Screen Type Decision Guide
|
||
|
||
```mermaid
|
||
graph TD
|
||
START["Need a screen?"] --> Q1{"Need specific\nresolution/depth?"}
|
||
Q1 -->|No| Q2{"Other apps should\nshare this screen?"}
|
||
Q1 -->|Yes| CUSTOM["CUSTOMSCREEN\nOwn resolution, palette, lifetime"]
|
||
Q2 -->|Yes| PUBLIC["PUBLICSCREEN\nNamed, discoverable by other apps"]
|
||
Q2 -->|No| WB["WBENCHSCREEN\nUse Workbench, no extra memory"]
|
||
|
||
CUSTOM --> Q3{"Need true-color\nor > 256 colors?"}
|
||
Q3 -->|Yes| RTG["RTG mode\nCyberGraphX / Picasso96"]
|
||
Q3 -->|No| NATIVE["Native chipset mode\nBestModeID() to pick"]
|
||
|
||
style WB fill:#e8f5e9,stroke:#4caf50,color:#333
|
||
style PUBLIC fill:#e8f4fd,stroke:#2196f3,color:#333
|
||
style CUSTOM fill:#fff3e0,stroke:#ff9800,color:#333
|
||
style RTG fill:#fce4ec,stroke:#e91e63,color:#333
|
||
```
|
||
|
||
| Choice | Memory Cost | Flexibility | When to Use |
|
||
|--------|------------|-------------|-------------|
|
||
| **Workbench** | None (shared) | Constrained to user's WB mode | Tools, utilities, simple apps |
|
||
| **Public Screen** | ~200–600 KB | Moderate — fixed at creation | Multi-app suites (MUI, email + browser) |
|
||
| **Custom Screen** | ~200–600 KB | Full — you own everything | Games, paint programs, video tools |
|
||
| **RTG Custom** | 1–4 MB | Full + true color | High-end graphics on gfx-card systems |
|
||
|
||
---
|
||
|
||
## Named Antipatterns
|
||
|
||
### "The Orphan Screen" — Leaking a Custom Screen
|
||
|
||
```c
|
||
/* BAD: Opening a custom screen but losing the pointer.
|
||
The screen stays open, consuming hundreds of KB of chip RAM,
|
||
and there's no way to close it without resetting. */
|
||
void ShowSplash(void)
|
||
{
|
||
OpenScreenTags(NULL,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Depth, 5,
|
||
SA_Title, "Splash",
|
||
TAG_DONE);
|
||
/* No pointer saved — screen is now a memory leak! */
|
||
}
|
||
```
|
||
|
||
```c
|
||
/* CORRECT: Always store and eventually close */
|
||
struct Screen *splashScr = NULL;
|
||
|
||
void ShowSplash(void)
|
||
{
|
||
splashScr = OpenScreenTags(NULL,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Depth, 5,
|
||
SA_Title, "Splash",
|
||
TAG_DONE);
|
||
}
|
||
|
||
void HideSplash(void)
|
||
{
|
||
if (splashScr)
|
||
{
|
||
CloseScreen(splashScr);
|
||
splashScr = NULL;
|
||
}
|
||
}
|
||
```
|
||
|
||
### "The Hardcoded Mode" — Assuming Display IDs
|
||
|
||
```c
|
||
/* BAD: Hardcoded PAL HiRes display ID.
|
||
On an NTSC Amiga, or one without ECS, this mode may not exist.
|
||
OpenScreenTags returns NULL — silent failure. */
|
||
struct Screen *scr = OpenScreenTags(NULL,
|
||
SA_DisplayID, 0x00029000, /* PAL HiRes — may not exist */
|
||
SA_Width, 640,
|
||
SA_Height, 256,
|
||
SA_Depth, 4,
|
||
TAG_DONE);
|
||
```
|
||
|
||
```c
|
||
/* CORRECT: Query the display database */
|
||
ULONG modeID = BestModeID(
|
||
BIDTAG_NominalWidth, 640,
|
||
BIDTAG_NominalHeight, 256,
|
||
BIDTAG_Depth, 4,
|
||
TAG_DONE);
|
||
|
||
if (modeID == INVALID_ID)
|
||
{
|
||
/* No suitable mode — fall back to Workbench */
|
||
modeID = INVALID_ID; /* OpenScreen on WB */
|
||
}
|
||
|
||
struct Screen *scr = OpenScreenTags(NULL,
|
||
SA_DisplayID, modeID,
|
||
SA_Width, 640,
|
||
SA_Height, 256,
|
||
SA_Depth, 4,
|
||
SA_Type, CUSTOMSCREEN,
|
||
TAG_DONE);
|
||
```
|
||
|
||
### "The Stale Pointer" — Using Screen After Close
|
||
|
||
```c
|
||
/* BAD: Caching a public screen pointer across operations.
|
||
If the screen owner closes it between your Lock and use,
|
||
you dereference freed memory — crash. */
|
||
struct Screen *scr = LockPubScreen("MYAPP");
|
||
UnlockPubScreen(NULL, scr);
|
||
/* ... 100ms passes, owner closes "MYAPP" screen ... */
|
||
DrawSomething(scr->RastPort); /* USE AFTER FREE */
|
||
```
|
||
|
||
```c
|
||
/* CORRECT: Re-lock immediately before each use */
|
||
struct Screen *scr = LockPubScreen("MYAPP");
|
||
if (scr)
|
||
{
|
||
DrawSomething(&scr->RastPort);
|
||
UnlockPubScreen(NULL, scr);
|
||
}
|
||
```
|
||
|
||
### "The Forced Close" — CloseScreen with Windows Open
|
||
|
||
```c
|
||
/* BAD: Closing a screen while visitor windows still exist.
|
||
CloseScreen() returns FALSE — the screen stays open.
|
||
But you've already freed your own window — confusion. */
|
||
CloseWindow(myWin);
|
||
CloseScreen(myScreen); /* May return FALSE — visitors still here */
|
||
```
|
||
|
||
```c
|
||
/* CORRECT: For public screens, go private first */
|
||
PubScreenStatus(myScreen, PSNF_PRIVATE); /* block new visitors */
|
||
|
||
/* Wait for remaining visitor windows to close */
|
||
while (myScreen->FirstWindow)
|
||
Delay(10);
|
||
|
||
CloseScreen(myScreen); /* now guaranteed to succeed */
|
||
```
|
||
|
||
### "The Layer Lock Trap" — Calling Intuition Inside LockLayerInfo
|
||
|
||
```c
|
||
/* BAD: Locking LayerInfo then calling any Intuition function.
|
||
Intuition needs LayerInfo internally — instant deadlock. */
|
||
LockLayerInfo(&scr->LayerInfo);
|
||
RefreshWindowFrame(win); /* DEADLOCK */
|
||
UnlockLayerInfo(&scr->LayerInfo);
|
||
```
|
||
|
||
```c
|
||
/* CORRECT: Never hold LayerInfo across Intuition calls.
|
||
If you need layer info, copy what you need, unlock, then proceed. */
|
||
LONG clipX = scr->Width; /* just read directly — it's atomic */
|
||
LONG clipY = scr->Height;
|
||
/* No lock needed for simple reads */
|
||
```
|
||
|
||
---
|
||
|
||
## Practical Cookbooks
|
||
|
||
### Cookbook 1: Screen Flipping (Game Double-Buffer)
|
||
|
||
Games often flip between two screens — the visible one and the hidden one being rendered:
|
||
|
||
```c
|
||
#include <proto/graphics.h>
|
||
#include <proto/intuition.h>
|
||
|
||
struct Screen *gameScreens[2];
|
||
int currentScreen = 0;
|
||
|
||
void InitGameScreens(void)
|
||
{
|
||
ULONG modeID = BestModeID(
|
||
BIDTAG_NominalWidth, 320,
|
||
BIDTAG_NominalHeight, 256,
|
||
BIDTAG_Depth, 5, /* 32 colors */
|
||
TAG_DONE);
|
||
|
||
for (int i = 0; i < 2; i++)
|
||
{
|
||
gameScreens[i] = OpenScreenTags(NULL,
|
||
SA_Width, 320,
|
||
SA_Height, 256,
|
||
SA_Depth, 5,
|
||
SA_DisplayID, modeID,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Quiet, TRUE, /* no title bar — full screen */
|
||
SA_Behind, (i == 1), /* second screen opens behind */
|
||
TAG_DONE);
|
||
}
|
||
}
|
||
|
||
void FlipScreen(void)
|
||
{
|
||
currentScreen = 1 - currentScreen;
|
||
ScreenToFront(gameScreens[currentScreen]);
|
||
}
|
||
|
||
struct Screen *GetRenderScreen(void)
|
||
{
|
||
return gameScreens[1 - currentScreen]; /* the hidden one */
|
||
}
|
||
|
||
void CleanupGameScreens(void)
|
||
{
|
||
for (int i = 0; i < 2; i++)
|
||
CloseScreen(gameScreens[i]);
|
||
}
|
||
```
|
||
|
||
> [!NOTE]
|
||
> `ScreenToFront()` / `ScreenToBack()` work on the screen depth arrangement, not the Copper list. Both screens are always being displayed — only the stacking order changes.
|
||
|
||
### Cookbook 2: Borderless Fullscreen Application
|
||
|
||
```c
|
||
/* Full-screen utility or game — no drag bar, no system gadgets,
|
||
every pixel belongs to the application. */
|
||
struct Screen *scr = OpenScreenTags(NULL,
|
||
SA_Width, 320,
|
||
SA_Height, 256,
|
||
SA_Depth, 5,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Quiet, TRUE, /* no title bar */
|
||
SA_ShowTitle, FALSE, /* redundant with SA_Quiet, but explicit */
|
||
SA_Pens, (ULONG)~0,
|
||
TAG_DONE);
|
||
|
||
/* Full-screen window (no borders, fills entire screen) */
|
||
struct Window *win = OpenWindowTags(NULL,
|
||
WA_CustomScreen, scr,
|
||
WA_Left, 0,
|
||
WA_Top, 0,
|
||
WA_Width, scr->Width,
|
||
WA_Height, scr->Height,
|
||
WA_Borderless, TRUE,
|
||
WA_Activate, TRUE,
|
||
WA_RMBTrap, TRUE, /* we handle right-click ourselves */
|
||
WA_IDCMP, IDCMP_RAWKEY | IDCMP_MOUSEBUTTONS |
|
||
IDCMP_MOUSEMOVE,
|
||
WA_Backdrop, TRUE, /* behind title bar area (irrelevant here) */
|
||
TAG_DONE);
|
||
|
||
/* Use scr->RastPort or win->RPort for full-screen drawing */
|
||
```
|
||
|
||
### Cookbook 3: PAL/NTSC Safe Screen Opening
|
||
|
||
```c
|
||
/* Open a screen that works on both PAL and NTSC machines.
|
||
Strategy: query the display database for the best available mode. */
|
||
|
||
#include <proto/graphics.h>
|
||
#include <graphics/displayinfo.h>
|
||
|
||
struct Screen *OpenSafeScreen(ULONG width, ULONG height, ULONG depth)
|
||
{
|
||
/* Step 1: Find the best matching mode */
|
||
ULONG modeID = BestModeID(
|
||
BIDTAG_NominalWidth, width,
|
||
BIDTAG_NominalHeight, height,
|
||
BIDTAG_Depth, depth,
|
||
TAG_DONE);
|
||
|
||
if (modeID == INVALID_ID)
|
||
{
|
||
/* Fallback: try the Workbench screen's mode */
|
||
struct Screen *wb = LockPubScreen(NULL);
|
||
if (wb)
|
||
{
|
||
modeID = GetVPModeID(&wb->ViewPort);
|
||
UnlockPubScreen(NULL, wb);
|
||
}
|
||
}
|
||
|
||
if (modeID == INVALID_ID)
|
||
return NULL; /* no usable display mode at all */
|
||
|
||
/* Step 2: Query actual dimensions for this mode */
|
||
struct DimensionInfo di;
|
||
if (GetDisplayInfoData(NULL, (UBYTE *)&di, sizeof(di),
|
||
DTAG_DIMS, modeID))
|
||
{
|
||
/* Use the mode's actual dimensions, not our requested ones */
|
||
return OpenScreenTags(NULL,
|
||
SA_DisplayID, modeID,
|
||
SA_Width, di.Nominal.MaxX - di.Nominal.MinX + 1,
|
||
SA_Height, di.Nominal.MaxY - di.Nominal.MinY + 1,
|
||
SA_Depth, depth,
|
||
SA_Type, CUSTOMSCREEN,
|
||
SA_Pens, (ULONG)~0,
|
||
TAG_DONE);
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
```
|
||
|
||
---
|
||
|
||
## Historical Context & Modern Analogies
|
||
|
||
### Competitive Landscape
|
||
|
||
| Platform | Multiple Resolutions | Screen Dragging | Independent Palettes | Notes |
|
||
|----------|---------------------|-----------------|---------------------|-------|
|
||
| **AmigaOS** | Yes — each screen independent | Yes — real-time drag | Yes — per-screen palette | Unique — no other platform does this |
|
||
| **Mac OS (Classic)** | No — one resolution globally | N/A | No — one global palette | Switchable via Monitors control panel (reboot) |
|
||
| **Windows 3.x** | No — one resolution globally | N/A | No — 16-color global palette | ALT+TAB switches between apps, not screens |
|
||
| **Atari ST** | No — fixed 640×400 (mono) or 320×200 (color) | N/A | No — fixed ST palette | Resolution requires hardware dip switches |
|
||
| **X11** | Yes — per-screen (Xinerama/XRandR) | No — virtual desktop instead | Yes — per-colormap | "Screen" means X screen, not Amiga screen |
|
||
|
||
The Amiga's multi-resolution coexistence was possible because the Copper could change display parameters mid-frame at zero CPU cost. Every other platform required the entire display to use one mode.
|
||
|
||
**Screen dragging** was Amiga-exclusive. The Mac had "window shuffling" (front/back), Windows had ALT+TAB, and X11 had virtual desktops — but none allowed dragging one resolution's display area to reveal a completely different resolution underneath, in real time.
|
||
|
||
### Modern Analogies
|
||
|
||
| Amiga Concept | Modern Equivalent | Notes |
|
||
|--------------|-------------------|-------|
|
||
| `OpenScreenTags()` | macOS `NSScreen` + `NSWindow` / X11 `XCreateWindow` | Amiga screens are independent display contexts |
|
||
| `CUSTOMSCREEN` | Fullscreen exclusive mode (DirectX / Vulkan) | Application owns the entire display |
|
||
| `PUBLICSCREEN` | Shared desktop / virtual desktop | Multiple apps share one display surface |
|
||
| `WBENCHSCREEN` | Primary display / main monitor | Default shared screen |
|
||
| Screen dragging | Virtual desktop panning / Spaces (macOS) | Amiga: physical drag; Modern: key-combo switch |
|
||
| `ScreenToFront()` / `ScreenToBack()` | `SetForegroundWindow()` / `NSWindow.orderFront` | Depth arrangement |
|
||
| Copper mid-frame mode switch | Not possible on modern GPUs | GPUs scan from a single framebuffer |
|
||
| `BestModeID()` | `EnumDisplaySettings()` / `CGDisplayAvailableModes()` | Query available modes |
|
||
| `LockPubScreen()` | `CGWindowListCopyWindowInfo()` / X11 `_NET_CLIENT_LIST` | Discover shared screens |
|
||
| `ViewPort` + `ColorMap` | `CRTC` + `LUT` on VGA / modern monitor | Hardware display timing |
|
||
| `SA_Overscan` | Overscan / underscan in GPU control panel | Edge-of-screen area usage |
|
||
| Interlaced flicker | N/A — modern displays are progressive | Amiga-specific CRT artifact |
|
||
|
||
---
|
||
|
||
## Use Cases
|
||
|
||
| Application | Screen Type | Resolution | Notable Pattern |
|
||
|-------------|------------|------------|-----------------|
|
||
| **Workbench** | WBENCHSCREEN | User-configured (default HiRes) | Persistent system screen |
|
||
| **Games (most)** | CUSTOMSCREEN, SA_Quiet | LoRes 320×256, 4–5 bitplanes | Borderless fullscreen, no drag bar |
|
||
| **Deluxe Paint** | CUSTOMSCREEN | LoRes or HAM depending on mode | Own palette, own resolution, screen dragging to access WB |
|
||
| **Word processors** | WBENCHSCREEN or PUBLICSCREEN | HiRes interlaced 640×512 | Share Workbench to save chip RAM |
|
||
| **Video tools (OPALVision, Video Toaster)** | CUSTOMSCREEN + genlock | Specific to video standard (NTSC/PAL) | Genlock overlay compositing |
|
||
| **DTP (PageStream)** | CUSTOMSCREEN, SA_AutoScroll | HiRes interlaced with virtual scroll | Screen larger than display, autoscroll |
|
||
| **MUI applications** | PUBLICSCREEN | User-configured | Named public screen shared by MUI apps |
|
||
| **Terminal emulators** | WBENCHSCREEN | HiRes 640×256+ | Open on Workbench to multitask |
|
||
|
||
---
|
||
|
||
## FAQ
|
||
|
||
**Q: Can I have two screens at different color depths on screen at the same time?**
|
||
A: Yes — this is one of the Amiga's signature features. Each screen has its own ViewPort with its own ColorMap. The Copper switches palette registers mid-frame. You can have a 4-color Workbench screen above a 32-color game screen, both fully visible.
|
||
|
||
**Q: How much chip RAM does a screen consume?**
|
||
A: It depends on resolution and depth. A 640×256 4-bitplane screen uses 640 × 256 × 4 / 8 = 81,920 bytes for bitplanes, plus ~5–10 KB for structures, color tables, and the Copper list. A 320×256 5-bitplane screen uses ~51,200 bytes. Interlaced screens double the height, doubling the cost.
|
||
|
||
**Q: Can I prevent the user from dragging my custom screen?**
|
||
A: Not easily. Screen dragging is handled by Intuition at a level above applications. Setting `SA_Quiet` removes the title bar (making it harder to grab), but Left-Amiga + mouse drag still works. Games typically just handle `IDCMP_CHANGEWINDOW` / `IDCMP_SCREENPOSITION` to detect and respond to unexpected moves.
|
||
|
||
**Q: What happens to my window when the user drags my screen behind another?**
|
||
A: Your window continues to exist and receive events normally. However, it's no longer visible. Keyboard events still route to the active window on your screen. Mouse events route to whichever screen the pointer is over.
|
||
|
||
**Q: Can I open a window on a screen owned by another process?**
|
||
A: Yes — if the screen is a public screen (opened with `SA_PubName`). Use `LockPubScreen("name")` to get the screen pointer, open your window with `WA_PubScreen`, then `UnlockPubScreen()`. You cannot open windows on another process's `CUSTOMSCREEN`.
|
||
|
||
**Q: How do I enumerate all open screens?**
|
||
A: Walk the `IntuitionBase->FirstScreen` linked list. Each screen's `NextScreen` field points to the next. You must lock Intuition with `LockIBase()` / `UnlockIBase()` to prevent the list from changing while you traverse it.
|
||
|
||
**Q: Why does `BestModeID()` return `INVALID_ID` on my A500?**
|
||
A: `BestModeID()` requires ECS or later to support queries beyond the basic modes. On OCS machines, use the known OCS mode constants (`LORES_KEY`, `HIRES_KEY`, etc.) directly, or fall back to `WBENCHSCREEN` if the query fails.
|
||
|
||
---
|
||
|
||
## Best Practices
|
||
|
||
1. **Prefer public screens** over custom screens — saves memory and lets users consolidate
|
||
2. **Always use `SA_Pens`** with `(ULONG)~0` to get proper 3D look under OS 2.0+
|
||
3. **Query the display database** with `BestModeID()` rather than hardcoding resolutions
|
||
4. **Handle `CloseScreen()` failure** — visitor windows may still be open
|
||
5. **Use `SA_AutoScroll`** for screens larger than the display — common for interlaced DTP work
|
||
6. **Lock/Unlock public screens properly** — `LockPubScreen()` prevents the screen from closing while you reference it
|
||
7. **Reply all IDCMP messages** — even during shutdown; drain the port before closing windows
|
||
8. **Never hold layer locks** while calling Intuition functions
|
||
9. **Use Right-Amiga** for application shortcuts — Left-Amiga is reserved for system use
|
||
10. **Test with multiple screens** at different resolutions to verify your input handling works correctly
|
||
11. **Use `SA_Quiet` for games** — removes title bar for immersive fullscreen
|
||
12. **Always check `OpenScreenTags()` return** — `NULL` means the mode is unavailable or memory is exhausted
|
||
|
||
---
|
||
|
||
## References
|
||
|
||
### NDK Headers
|
||
|
||
- `intuition/screens.h` — `struct Screen`, `SA_*` tag definitions, `CUSTOMSCREEN`/`PUBLICSCREEN`/`WBENCHSCREEN`
|
||
- `intuition/intuition.h` — `struct Window`, `WA_*` tags, IDCMP flags
|
||
- `graphics/displayinfo.h` — Display IDs, `DimensionInfo`, `BestModeID()`
|
||
- `graphics/gfxbase.h` — `GfxBase->DisplayFirst` for display database traversal
|
||
|
||
### Autodocs
|
||
|
||
- ADCD 2.1: `OpenScreenTagList()`, `CloseScreen()`, `ScreenToFront()`, `ScreenToBack()`
|
||
- ADCD 2.1: `LockPubScreen()`, `UnlockPubScreen()`, `PubScreenStatus()`, `LockIBase()`
|
||
- ADCD 2.1: `BestModeID()`, `NextDisplayInfo()`, `GetDisplayInfoData()`
|
||
- ADCD 2.1: `GetScreenDrawInfo()`, `FreeScreenDrawInfo()`, `GetVPModeID()`
|
||
|
||
### Related Knowledge Base Articles
|
||
|
||
- [Copper](../08_graphics/copper/copper.md) — the hardware mechanism that makes multi-screen possible
|
||
- [Copper Programming](../08_graphics/copper/copper_programming.md) — writing Copper lists
|
||
- [Views](../08_graphics/views.md) — View/ViewPort/ViewPortExtra hierarchy
|
||
- [Bitmap](../08_graphics/bitmap.md) — screen bitmap layout and bitplane organization
|
||
- [Windows](windows.md) — windows live on screens
|
||
- [IDCMP](idcmp.md) — input event delivery to windows
|
||
- [Display Modes](../08_graphics/display_modes.md) — resolution, color depth, and special modes
|
||
- [RTG Programming](../08_graphics/rtg_programming.md) — graphics card screens
|