123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489 |
- #import win32 "sys/windows.odin";
- #import "fmt.odin";
- #import "os.odin";
- #import "mem.odin";
- CANVAS_WIDTH :: 128;
- CANVAS_HEIGHT :: 128;
- CANVAS_SCALE :: 3;
- FRAME_TIME :: 1.0/30.0;
- WINDOW_TITLE :: "Punity\x00";
- _ := compile_assert(CANVAS_WIDTH % 16 == 0);
- WINDOW_WIDTH :: CANVAS_WIDTH * CANVAS_SCALE;
- WINDOW_HEIGHT :: CANVAS_HEIGHT * CANVAS_SCALE;
- STACK_CAPACITY :: 1<<20;
- STORAGE_CAPACITY :: 1<<20;
- DRAW_LIST_RESERVE :: 128;
- MAX_KEYS :: 256;
- Core :: struct {
- stack: ^Bank,
- storage: ^Bank,
- running: bool,
- key_modifiers: u32,
- key_states: [MAX_KEYS]byte,
- key_deltas: [MAX_KEYS]byte,
- perf_frame,
- perf_frame_inner,
- perf_step,
- perf_audio,
- perf_blit,
- perf_blit_cvt,
- perf_blit_gdi: Perf_Span,
- frame: i64,
- canvas: Canvas,
- draw_list: ^Draw_List,
- }
- Perf_Span :: struct {
- stamp: f64,
- delta: f32,
- }
- Bank :: struct {
- memory: []byte,
- cursor: int,
- }
- Bank_State :: struct {
- state: Bank,
- bank: ^Bank,
- }
- Color :: raw_union {
- using channels: struct{a, b, g, r: byte},
- rgba: u32,
- }
- Palette :: struct {
- colors: [256]Color,
- colors_count: byte,
- }
- Rect :: raw_union {
- using minmax: struct {min_x, min_y, max_x, max_y: int},
- using pos: struct {left, top, right, bottom: int},
- e: [4]int,
- }
- Bitmap :: struct {
- pixels: []byte,
- width: int,
- height: int,
- }
- Font :: struct {
- using bitmap: Bitmap,
- char_width: int,
- char_height: int,
- }
- Canvas :: struct {
- using bitmap: ^Bitmap,
- palette: Palette,
- translate_x: int,
- translate_y: int,
- clip: Rect,
- font: ^Font,
- }
- DrawFlag :: enum {
- NONE = 0,
- FLIP_H = 1<<0,
- FLIP_V = 1<<1,
- MASK = 1<<2,
- }
- Draw_Item :: struct {}
- Draw_List :: struct {
- items: []Draw_Item,
- }
- Key :: enum {
- MOD_SHIFT = 0x0001,
- MOD_CONTROL = 0x0002,
- MOD_ALT = 0x0004,
- MOD_SUPER = 0x0008,
- UNKNOWN =-1,
- INVALID =-2,
- LBUTTON = 1,
- RBUTTON = 2,
- CANCEL = 3,
- MBUTTON = 4,
- BACK = 8,
- TAB = 9,
- CLEAR = 12,
- RETURN = 13,
- SHIFT = 16,
- CONTROL = 17,
- MENU = 18,
- PAUSE = 19,
- CAPITAL = 20,
- KANA = 0x15,
- HANGEUL = 0x15,
- HANGUL = 0x15,
- JUNJA = 0x17,
- FINAL = 0x18,
- HANJA = 0x19,
- KANJI = 0x19,
- ESCAPE = 0x1B,
- CONVERT = 0x1C,
- NONCONVERT = 0x1D,
- ACCEPT = 0x1E,
- MODECHANGE = 0x1F,
- SPACE = 32,
- PRIOR = 33,
- NEXT = 34,
- END = 35,
- HOME = 36,
- LEFT = 37,
- UP = 38,
- RIGHT = 39,
- DOWN = 40,
- SELECT = 41,
- PRINT = 42,
- EXEC = 43,
- SNAPSHOT = 44,
- INSERT = 45,
- DELETE = 46,
- HELP = 47,
- LWIN = 0x5B,
- RWIN = 0x5C,
- APPS = 0x5D,
- SLEEP = 0x5F,
- NUMPAD0 = 0x60,
- NUMPAD1 = 0x61,
- NUMPAD2 = 0x62,
- NUMPAD3 = 0x63,
- NUMPAD4 = 0x64,
- NUMPAD5 = 0x65,
- NUMPAD6 = 0x66,
- NUMPAD7 = 0x67,
- NUMPAD8 = 0x68,
- NUMPAD9 = 0x69,
- MULTIPLY = 0x6A,
- ADD = 0x6B,
- SEPARATOR = 0x6C,
- SUBTRACT = 0x6D,
- DECIMAL = 0x6E,
- DIVIDE = 0x6F,
- F1 = 0x70,
- F2 = 0x71,
- F3 = 0x72,
- F4 = 0x73,
- F5 = 0x74,
- F6 = 0x75,
- F7 = 0x76,
- F8 = 0x77,
- F9 = 0x78,
- F10 = 0x79,
- F11 = 0x7A,
- F12 = 0x7B,
- F13 = 0x7C,
- F14 = 0x7D,
- F15 = 0x7E,
- F16 = 0x7F,
- F17 = 0x80,
- F18 = 0x81,
- F19 = 0x82,
- F20 = 0x83,
- F21 = 0x84,
- F22 = 0x85,
- F23 = 0x86,
- F24 = 0x87,
- NUMLOCK = 0x90,
- SCROLL = 0x91,
- LSHIFT = 0xA0,
- RSHIFT = 0xA1,
- LCONTROL = 0xA2,
- RCONTROL = 0xA3,
- LMENU = 0xA4,
- RMENU = 0xA5,
- APOSTROPHE = 39, /* ' */
- COMMA = 44, /* , */
- MINUS = 45, /* - */
- PERIOD = 46, /* . */
- SLASH = 47, /* / */
- NUM0 = 48,
- NUM1 = 49,
- NUM2 = 50,
- NUM3 = 51,
- NUM4 = 52,
- NUM5 = 53,
- NUM6 = 54,
- NUM7 = 55,
- NUM8 = 56,
- NUM9 = 57,
- SEMICOLON = 59, /* ; */
- EQUAL = 61, /* = */
- A = 65,
- B = 66,
- C = 67,
- D = 68,
- E = 69,
- F = 70,
- G = 71,
- H = 72,
- I = 73,
- J = 74,
- K = 75,
- L = 76,
- M = 77,
- N = 78,
- O = 79,
- P = 80,
- Q = 81,
- R = 82,
- S = 83,
- T = 84,
- U = 85,
- V = 86,
- W = 87,
- X = 88,
- Y = 89,
- Z = 90,
- LEFT_BRACKET = 91, /* [ */
- BACKSLASH = 92, /* \ */
- RIGHT_BRACKET = 93, /* ] */
- GRAVE_ACCENT = 96, /* ` */
- };
- key_down :: proc(k: Key) -> bool {
- return _core.key_states[k] != 0;
- }
- key_pressed :: proc(k: Key) -> bool {
- return (_core.key_deltas[k] != 0) && key_down(k);
- }
- win32_perf_count_freq := win32.GetQueryPerformanceFrequency();
- time_now :: proc() -> f64 {
- assert(win32_perf_count_freq != 0);
- counter: i64;
- win32.QueryPerformanceCounter(^counter);
- result := cast(f64)counter / cast(f64)win32_perf_count_freq;
- return result;
- }
- _core: Core;
- run :: proc(user_init, user_step: proc(c: ^Core)) {
- using win32;
- _core.running = true;
- win32_proc :: proc(hwnd: win32.HWND, msg: u32, wparam: win32.WPARAM, lparam: win32.LPARAM) -> win32.LRESULT #no_inline #cc_c {
- win32_app_key_mods :: proc() -> u32 {
- mods: u32 = 0;
- if is_key_down(Key_Code.SHIFT) {
- mods |= cast(u32)Key.MOD_SHIFT;
- }
- if is_key_down(Key_Code.CONTROL) {
- mods |= cast(u32)Key.MOD_CONTROL;
- }
- if is_key_down(Key_Code.MENU) {
- mods |= cast(u32)Key.MOD_ALT;
- }
- if is_key_down(Key_Code.LWIN) || is_key_down(Key_Code.RWIN) {
- mods |= cast(u32)Key.MOD_SUPER;
- }
- return mods;
- }
- match msg {
- case WM_KEYDOWN:
- _core.key_modifiers = win32_app_key_mods();
- if wparam < MAX_KEYS {
- _core.key_states[wparam] = 1;
- _core.key_deltas[wparam] = 1;
- }
- return 0;
- case WM_KEYUP:
- _core.key_modifiers = win32_app_key_mods();
- if wparam < MAX_KEYS {
- _core.key_states[wparam] = 0;
- _core.key_deltas[wparam] = 1;
- }
- return 0;
- case WM_CLOSE:
- PostQuitMessage(0);
- _core.running = false;
- return 0;
- }
- return DefWindowProcA(hwnd, msg, wparam, lparam);
- }
- window_class := WNDCLASSEXA{
- class_name = (cast(string)"Punity\x00").data, // C-style string
- size = size_of(WNDCLASSEXA),
- style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC,
- instance = cast(HINSTANCE)GetModuleHandleA(nil),
- wnd_proc = win32_proc,
- // wnd_proc = DefWindowProcA,
- background = cast(HBRUSH)GetStockObject(BLACK_BRUSH),
- };
- if RegisterClassExA(^window_class) == 0 {
- fmt.fprintln(os.stderr, "RegisterClassExA failed");
- return;
- }
- screen_width := GetSystemMetrics(SM_CXSCREEN);
- screen_height := GetSystemMetrics(SM_CYSCREEN);
- rc: RECT;
- rc.left = (screen_width - WINDOW_WIDTH) / 2;
- rc.top = (screen_height - WINDOW_HEIGHT) / 2;
- rc.right = rc.left + WINDOW_WIDTH;
- rc.bottom = rc.top + WINDOW_HEIGHT;
- style: u32 = WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX;
- assert(AdjustWindowRect(^rc, style, 0) != 0);
- wt := WINDOW_TITLE;
- win32_window := CreateWindowExA(0,
- window_class.class_name,
- wt.data,
- style,
- rc.left, rc.top,
- rc.right-rc.left, rc.bottom-rc.top,
- nil, nil, window_class.instance,
- nil);
- if win32_window == nil {
- fmt.fprintln(os.stderr, "CreateWindowExA failed");
- return;
- }
- window_bmi: BITMAPINFO;
- window_bmi.size = size_of(BITMAPINFOHEADER);
- window_bmi.width = CANVAS_WIDTH;
- window_bmi.height = CANVAS_HEIGHT;
- window_bmi.planes = 1;
- window_bmi.bit_count = 32;
- window_bmi.compression = BI_RGB;
- user_init(^_core);
- ShowWindow(win32_window, SW_SHOW);
- window_buffer := new_slice(u32, CANVAS_WIDTH * CANVAS_HEIGHT);
- defer free(window_buffer);
- for i := 0; i < window_buffer.count; i += 1 {
- window_buffer[i] = 0xff00ff;
- }
- dt: f64;
- prev_time := time_now();
- curr_time := time_now();
- total_time : f64 = 0;
- offset_x := 0;
- offset_y := 0;
- message: MSG;
- for _core.running {
- curr_time = time_now();
- dt = curr_time - prev_time;
- prev_time = curr_time;
- total_time += dt;
- offset_x += 1;
- offset_y += 2;
- {
- data: [128]byte;
- buf: fmt.Buffer;
- buf.data = data[:];
- fmt.bprintf(^buf, "Punity: %.4f ms\x00", dt*1000);
- win32.SetWindowTextA(win32_window, ^buf[0]);
- }
- for y := 0; y < CANVAS_HEIGHT; y += 1 {
- for x := 0; x < CANVAS_WIDTH; x += 1 {
- g := (x % 32) * 8;
- b := (y % 32) * 8;
- window_buffer[x + y*CANVAS_WIDTH] = cast(u32)(g << 8 | b);
- }
- }
- mem.zero(^_core.key_deltas[0], size_of_val(_core.key_deltas));
- for PeekMessageA(^message, nil, 0, 0, PM_REMOVE) != 0 {
- if message.message == WM_QUIT {
- _core.running = false;
- }
- TranslateMessage(^message);
- DispatchMessageA(^message);
- }
- user_step(^_core);
- dc := GetDC(win32_window);
- StretchDIBits(dc,
- 0, 0, CANVAS_WIDTH * CANVAS_SCALE, CANVAS_HEIGHT * CANVAS_SCALE,
- 0, 0, CANVAS_WIDTH, CANVAS_HEIGHT,
- window_buffer.data,
- ^window_bmi,
- DIB_RGB_COLORS,
- SRCCOPY);
- ReleaseDC(win32_window, dc);
- {
- delta := time_now() - prev_time;
- ms := cast(i32)((FRAME_TIME - delta) * 1000);
- if ms > 0 {
- win32.Sleep(ms);
- }
- }
- _core.frame += 1;
- }
- }
- main :: proc() {
- user_init :: proc(c: ^Core) {
- }
- user_step :: proc(c: ^Core) {
- }
- run(user_init, user_step);
- }
|