Compare commits
	
		
			9 Commits 
		
	
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | ba620e66aa | |
|  | 12d2d7263e | |
|  | 8cff4fb8a6 | |
|  | fb60bd5f32 | |
|  | 89f0a15ade | |
|  | 85df8265b0 | |
|  | 2942efe6e2 | |
|  | 6119d34559 | |
|  | 4c1d4f7547 | 
|  | @ -11,8 +11,8 @@ | ||||||
|       "bin/an_editor.o", |       "bin/an_editor.o", | ||||||
|       "src/main.c" |       "src/main.c" | ||||||
|     ], |     ], | ||||||
|     "directory": "/home/patrick/Documents/an_editor", |     "directory": "/home/patrick/Documents/a_chat_client", | ||||||
|     "file": "/home/patrick/Documents/an_editor/src/main.c", |     "file": "/home/patrick/Documents/a_chat_client/src/main.c", | ||||||
|     "output": "/home/patrick/Documents/an_editor/bin/an_editor.o" |     "output": "/home/patrick/Documents/a_chat_client/bin/a_chat_client" | ||||||
|   } |   } | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  | @ -37,6 +37,7 @@ | ||||||
|             pkg-config |             pkg-config | ||||||
|             binutils |             binutils | ||||||
|             curlMinimal |             curlMinimal | ||||||
|  |             curl-config | ||||||
|             clang |             clang | ||||||
|             bear |             bear | ||||||
|             naga-cli |             naga-cli | ||||||
|  | @ -48,12 +49,14 @@ | ||||||
|           ] else if pkgs.system == "x86_64-linux" then [ |           ] else if pkgs.system == "x86_64-linux" then [ | ||||||
|             pkg-config |             pkg-config | ||||||
|             binutils |             binutils | ||||||
|  |             curlMinimal | ||||||
|             clang |             clang | ||||||
|             bear |             bear | ||||||
|             naga-cli |             naga-cli | ||||||
|             libGL |             libGL | ||||||
|             mesa |             mesa | ||||||
|             gf |             gf | ||||||
|  |             wayland | ||||||
|             xorg.libX11 |             xorg.libX11 | ||||||
|             xorg.libXi |             xorg.libXi | ||||||
|             xorg.xinput |             xorg.xinput | ||||||
|  |  | ||||||
							
								
								
									
										28
									
								
								justfile
								
								
								
								
							
							
						
						
									
										28
									
								
								justfile
								
								
								
								
							|  | @ -1,16 +1,36 @@ | ||||||
| alias b := build | alias b := build | ||||||
| alias r := run | alias r := run | ||||||
| 
 | 
 | ||||||
|  | c_flags := if os() == "macos" { | ||||||
|  |     "$(curl-config --libs) -framework Cocoa -framework QuartzCore -framework CoreImage -framework Metal -framework MetalKit -ObjC" | ||||||
|  | } else if os_family() == "unix" { | ||||||
|  |     "$(curl-config --libs) -lEGL -lGLESv2 -lGL -lm -lwayland-client -lwayland-egl -lX11 -lXi -lXcursor -Wno-implicit-function-declaration" | ||||||
|  | } else { "" } | ||||||
|  | 
 | ||||||
|  | [macos] | ||||||
| build: transpile_shaders_metal | build: transpile_shaders_metal | ||||||
|     mkdir -p bin |     mkdir -p bin | ||||||
|     cc -Ivendor/ -O0 -g -Wall -Wextra $(curl-config --libs) -framework Cocoa -framework QuartzCore -framework CoreImage -framework Metal -framework MetalKit -ObjC src/*.c -o bin/chat_client |     cc -Ivendor/ -O0 -g -Wall -Wextra {{ c_flags }} src/*.c -o bin/chat_client | ||||||
|     # cc -Ivendor/ -g -Wall -Wextra src/*.c -o bin/chat_client -lEGL -lGLESv2 -lGL -lm -lX11 -lXi -lXcursor |  | ||||||
|     # cc bin/*.o -o bin/chat_client -lEGL -lGLESv2 -lGL -lm -lX11 -lXi -lXcursor |  | ||||||
| 
 | 
 | ||||||
|  | [linux] | ||||||
|  | build: generate_wayland_protocols | ||||||
|  |     mkdir -p bin | ||||||
|  |     cc -Ivendor/ -O0 -g -Wall -Wextra {{ c_flags }} src/*.c src/wayland-crap/*.c -o bin/chat_client | ||||||
|  | 
 | ||||||
|  | [linux] | ||||||
|  | run: build | ||||||
|  |     nixGLIntel ./bin/chat_client | ||||||
|  | 
 | ||||||
|  | [macos] | ||||||
| run: build | run: build | ||||||
|     # nixGLIntel ./bin/chat_client |  | ||||||
|     ./bin/chat_client |     ./bin/chat_client | ||||||
| 
 | 
 | ||||||
|  | [linux] | ||||||
|  | generate_wayland_protocols: | ||||||
|  |     wayland-scanner client-header src/wayland-crap/xdg-shell.xml src/wayland-crap/xdg-shell.h | ||||||
|  |     wayland-scanner code src/wayland-crap/xdg-shell.xml src/wayland-crap/xdg-shell.c | ||||||
|  | 
 | ||||||
|  | [macos] | ||||||
| transpile_shaders_metal: | transpile_shaders_metal: | ||||||
|     mkdir -p bin/transpiled_shaders |     mkdir -p bin/transpiled_shaders | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,18 @@ | ||||||
|  | #version 440 core | ||||||
|  | 
 | ||||||
|  | struct VertexOutput { | ||||||
|  |     vec4 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | layout(location = 0) in VertexOutput out_vertex; | ||||||
|  | layout(location = 1) uniform highp sampler2D atlas_texture; | ||||||
|  | 
 | ||||||
|  | out vec4 color; | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     float text_color = texture(atlas_texture, out_vertex.tex_coord).r; | ||||||
|  | 
 | ||||||
|  |     color = vec4(text_color * vec3(1,1,1), text_color); | ||||||
|  | } | ||||||
|  | @ -0,0 +1,56 @@ | ||||||
|  | #version 440 core | ||||||
|  | 
 | ||||||
|  | struct Vertex { | ||||||
|  |     vec2 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct VertexOutput { | ||||||
|  |     vec4 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct Glyph { | ||||||
|  |     vec2 atlas_position; | ||||||
|  |     vec2 size; | ||||||
|  |     vec2 target_position; | ||||||
|  |     float y_offset; | ||||||
|  |     float _haha_alignment; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | layout(std430, binding = 0) readonly buffer VertexBlock { | ||||||
|  |     Vertex vertices[]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | layout(std430, binding = 1) readonly buffer GlyphBlock { | ||||||
|  |     Glyph glyphs[]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | layout(std430, binding = 2) readonly buffer ParamsBlock { | ||||||
|  |     vec2 screen_size; | ||||||
|  |     vec2 font_size; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | vec4 to_device_position(vec2 position, vec2 size) { | ||||||
|  |     return vec4(((position / size) * 2.0) - vec2(1.0), 1.0, 1.0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | layout(location = 0) out VertexOutput out_vertex; | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     Glyph glyph = glyphs[gl_InstanceID]; | ||||||
|  | 
 | ||||||
|  |     vec2 scaled_size = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size/2.0); | ||||||
|  |     vec2 scaled_size_2 = ((vertices[gl_VertexID].position + 1.0) / 2.0) * (glyph.size); | ||||||
|  |     vec2 glyph_pos = scaled_size + glyph.target_position + vec2(0, glyph.y_offset/2.0+font_size.y); | ||||||
|  | 
 | ||||||
|  |     vec4 device_position = to_device_position(glyph_pos, screen_size); | ||||||
|  |     vec2 atlas_position = (scaled_size_2 + glyph.atlas_position) / 512.0; | ||||||
|  | 
 | ||||||
|  |     device_position.y = -device_position.y; | ||||||
|  | 
 | ||||||
|  |     out_vertex.position = device_position; | ||||||
|  |     out_vertex.tex_coord = atlas_position; | ||||||
|  | 
 | ||||||
|  |     gl_Position = device_position; | ||||||
|  | } | ||||||
|  | @ -0,0 +1,45 @@ | ||||||
|  | #version 440 core | ||||||
|  | 
 | ||||||
|  | struct UiRectFragment { | ||||||
|  |     highp vec4 device_position; | ||||||
|  |     highp vec2 position; | ||||||
|  |     highp vec2 size; | ||||||
|  |     highp vec2 border_size; | ||||||
|  |     highp vec2 screen_size; | ||||||
|  |     highp vec2 tex_coord; | ||||||
|  |     highp vec4 color; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | in UiRectFragment out_rect; | ||||||
|  | 
 | ||||||
|  | layout(location = 0) out highp vec4 color; | ||||||
|  | 
 | ||||||
|  | highp float rect_sdf( | ||||||
|  |     highp vec2 absolute_pixel_position, | ||||||
|  |     highp vec2 origin, | ||||||
|  |     highp vec2 size, | ||||||
|  |     highp float corner_radius | ||||||
|  | ) { | ||||||
|  |     highp vec2 half_size = size / 2.0; | ||||||
|  |     highp vec2 rect_center = origin + half_size; | ||||||
|  | 
 | ||||||
|  |     highp vec2 pixel_position = abs(absolute_pixel_position - rect_center); | ||||||
|  |     highp vec2 shrunk_corner_position = half_size - corner_radius; | ||||||
|  | 
 | ||||||
|  |     highp vec2 pixel_to_shrunk_corner = max(vec2(0), pixel_position - shrunk_corner_position); | ||||||
|  |     highp float distance_to_shrunk_corner = length(pixel_to_shrunk_corner); | ||||||
|  |     highp float distance = distance_to_shrunk_corner - corner_radius; | ||||||
|  | 
 | ||||||
|  |     return distance; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     highp vec2 pixel_pos = out_rect.tex_coord.xy * out_rect.screen_size; | ||||||
|  | 
 | ||||||
|  |     highp float distance = rect_sdf(pixel_pos, out_rect.position, out_rect.size, out_rect.border_size.x); | ||||||
|  |     if (distance <= 0.0) { | ||||||
|  |         color = out_rect.color; | ||||||
|  |     } else { | ||||||
|  |         color = vec4(0); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -0,0 +1,50 @@ | ||||||
|  | #version 440 core | ||||||
|  | 
 | ||||||
|  | struct Vertex { | ||||||
|  |     vec2 position; | ||||||
|  |     vec2 tex_coord; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct UiRect { | ||||||
|  |     vec4 position; | ||||||
|  |     vec4 size; | ||||||
|  |     vec4 border_size; | ||||||
|  |     vec4 color; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct UiRectFragment { | ||||||
|  |     highp vec4 device_position; | ||||||
|  |     highp vec2 position; | ||||||
|  |     highp vec2 size; | ||||||
|  |     highp vec2 border_size; | ||||||
|  |     highp vec2 screen_size; | ||||||
|  |     highp vec2 tex_coord; | ||||||
|  |     highp vec4 color; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | layout(std430, binding = 0) readonly buffer VertexBlock { | ||||||
|  |     Vertex vertices[]; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | layout(std430, binding = 1) readonly buffer RectBlock { | ||||||
|  |     UiRect rects[]; | ||||||
|  | }; | ||||||
|  | layout(std430, binding = 2) readonly buffer ParamsBlock { | ||||||
|  |     vec2 screen_size; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | out UiRectFragment out_rect; | ||||||
|  | 
 | ||||||
|  | void main() { | ||||||
|  |     UiRect rect = rects[gl_InstanceID]; | ||||||
|  | 
 | ||||||
|  |     out_rect.device_position = vec4(vertices[gl_VertexID].position, 1, 1); | ||||||
|  |     out_rect.position = rect.position.xy; | ||||||
|  |     out_rect.size = rect.size.xy; | ||||||
|  |     out_rect.border_size = rect.border_size.xy; | ||||||
|  |     out_rect.screen_size = screen_size; | ||||||
|  |     out_rect.tex_coord = vertices[gl_VertexID].tex_coord; | ||||||
|  |     out_rect.color = rect.color; | ||||||
|  | 
 | ||||||
|  |     gl_Position = vec4(vertices[gl_VertexID].position, 1, 1); | ||||||
|  | } | ||||||
							
								
								
									
										142
									
								
								src/ed_array.h
								
								
								
								
							
							
						
						
									
										142
									
								
								src/ed_array.h
								
								
								
								
							|  | @ -1,80 +1,90 @@ | ||||||
| #ifndef ED_ARRAY_INCLUDED | #ifndef ED_ARRAY_INCLUDED | ||||||
| #define ED_ARRAY_INCLUDED | #define ED_ARRAY_INCLUDED | ||||||
| #include <stdio.h> |  | ||||||
| #include <stdlib.h> |  | ||||||
| #include <assert.h> | #include <assert.h> | ||||||
| #include <memory.h> | #include <memory.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
| 
 | 
 | ||||||
| #define array(T) struct T ## _Array | #define array(T) struct T##_Array | ||||||
| #define arrayTemplate(T) array(T) {\ | #define arrayTemplate(T)                                                       \ | ||||||
|     size_t size, capacity;\ |     array(T) {                                                                 \ | ||||||
|     T *data;\ |         size_t size, capacity;                                                 \ | ||||||
| };\ |         T *data;                                                               \ | ||||||
| array(T) T ## _ArrayConstruct(size_t size) {\ |     };                                                                         \ | ||||||
|     array(T) this;\ |     array(T) T##_ArrayConstruct(size_t size) {                                 \ | ||||||
|     this.size = 0;\ |         array(T) this;                                                         \ | ||||||
|     this.capacity = size;\ |         this.size = 0;                                                         \ | ||||||
|     this.data = malloc(size * sizeof(T));\ |         this.capacity = size;                                                  \ | ||||||
|     if (!this.data) {\ |         this.data = malloc(size * sizeof(T));                                  \ | ||||||
|         assert("failed to allocate memory for array");\ |         if (!this.data) {                                                      \ | ||||||
|     }\ |             assert("failed to allocate memory for array");                     \ | ||||||
|     return this;\ |         }                                                                      \ | ||||||
| };\ |         return this;                                                           \ | ||||||
| void T ## _PushArray(array(T) *arr, T value) {\ |     };                                                                         \ | ||||||
|     if (arr->size+1 <= arr->capacity) {\ |     void T##_PushArray(array(T) * arr, T value) {                              \ | ||||||
|         arr->data[arr->size] = value;\ |         if (arr->size == arr->capacity) {                                      \ | ||||||
|         arr->size += 1;\ |             arr->capacity *= 2;                                                \ | ||||||
|     } else {\ |             void *new_data = realloc(arr->data, arr->capacity);                \ | ||||||
|         fprintf(stderr, "failed to push to u8 array, size+num > capacity\n");\ |             if (new_data == NULL) {                                            \ | ||||||
|     }\ |                 fprintf(stderr, "out of memory when reallocating array\n");    \ | ||||||
| };\ |             }                                                                  \ | ||||||
| void T ## _PushArrayMulti(array(T) *arr, T *values, size_t len) {\ |             arr->data = new_data;                                              \ | ||||||
|     for (size_t i = 0; i < len; ++i) {\ |         }                                                                      \ | ||||||
|         T ## _PushArray(arr, values[i]);\ |         if (arr->size + 1 <= arr->capacity) {                                  \ | ||||||
|     }\ |             arr->data[arr->size] = value;                                      \ | ||||||
| };\ |             arr->size += 1;                                                    \ | ||||||
| void T ## _InsertArrayAt(array(T) *arr, size_t loc, T value) {\ |         }                                                                      \ | ||||||
|     if (arr->size == arr->capacity) {\ |     };                                                                         \ | ||||||
|         arr->capacity *= 2;\ |     void T##_PushArrayMulti(array(T) * arr, T * values, size_t len) {          \ | ||||||
|         void *new_data = realloc(arr->data, arr->capacity);\ |         for (size_t i = 0; i < len; ++i) {                                     \ | ||||||
|         if (new_data == NULL) {\ |             T##_PushArray(arr, values[i]);                                     \ | ||||||
|             fprintf(stderr, "out of memory when reallocating array\n");\ |         }                                                                      \ | ||||||
|         }\ |     };                                                                         \ | ||||||
|         arr->data = new_data;\ |     void T##_InsertArrayAt(array(T) * arr, size_t loc, T value) {              \ | ||||||
|     }\ |         if (arr->size == arr->capacity) {                                      \ | ||||||
|     memcpy(&arr->data[loc+1], &arr->data[loc], arr->size * sizeof(T));\ |             arr->capacity *= 2;                                                \ | ||||||
|     arr->data[loc] = value;\ |             void *new_data = realloc(arr->data, arr->capacity);                \ | ||||||
|     arr->size += 1;\ |             if (new_data == NULL) {                                            \ | ||||||
| }; |                 fprintf(stderr, "out of memory when reallocating array\n");    \ | ||||||
|  |             }                                                                  \ | ||||||
|  |             arr->data = new_data;                                              \ | ||||||
|  |         }                                                                      \ | ||||||
|  |         memcpy(&arr->data[loc + 1], &arr->data[loc], arr->size * sizeof(T));   \ | ||||||
|  |         arr->data[loc] = value;                                                \ | ||||||
|  |         arr->size += 1;                                                        \ | ||||||
|  |     }; | ||||||
| 
 | 
 | ||||||
| #define slice(T) struct T ## _Slice | #define slice(T) struct T##_Slice | ||||||
| #define sliceTemplate(T) slice(T) {\ | #define sliceTemplate(T)                                                       \ | ||||||
|     size_t len;\ |     slice(T) {                                                                 \ | ||||||
|     T *data;\ |         size_t len;                                                            \ | ||||||
| };\ |         T *data;                                                               \ | ||||||
| array(T) T ## _FromSlice(slice(T) s) {\ |     };                                                                         \ | ||||||
|     array(T) arr = T ## _ArrayConstruct(s.len);\ |     array(T) T##_FromSlice(slice(T) s) {                                       \ | ||||||
|     memcpy(arr.data, s.data, sizeof(T) * s.len);\ |         array(T) arr = T##_ArrayConstruct(s.len);                              \ | ||||||
|     arr.size = s.len;\ |         memcpy(arr.data, s.data, sizeof(T) * s.len);                           \ | ||||||
|     return arr;\ |         arr.size = s.len;                                                      \ | ||||||
| }\ |         return arr;                                                            \ | ||||||
| slice(T) T ## _SliceConstruct(void *data, size_t len) {\ |     }                                                                          \ | ||||||
|     slice(T) this;\ |     slice(T) T##_SliceConstruct(void *data, size_t len) {                      \ | ||||||
|     this.len = len;\ |         slice(T) this;                                                         \ | ||||||
|     this.data = data;\ |         this.len = len;                                                        \ | ||||||
|     return this;\ |         this.data = data;                                                      \ | ||||||
| } |         return this;                                                           \ | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
| #define newArray(T, size) T ## _ArrayConstruct(size) | #define newArray(T, size) T##_ArrayConstruct(size) | ||||||
| #define newArrayFromSlice(T, s) T ## _FromSlice(s) | #define newArrayFromSlice(T, s) T##_FromSlice(s) | ||||||
| #define pushArray(T, arr, value) T ## _PushArray(arr, (value)) | #define pushArray(T, arr, value) T##_PushArray(arr, (value)) | ||||||
| #define pushArrayMulti(T, arr, values, len) T ## _PushArrayMulti(arr, values, len) | #define pushArrayMulti(T, arr, values, len) T##_PushArrayMulti(arr, values, len) | ||||||
| 
 | 
 | ||||||
| #define insertArrayAt(T, arr, loc, value) T ## _InsertArrayAt(arr, loc, (value)) | #define insertArrayAt(T, arr, loc, value) T##_InsertArrayAt(arr, loc, (value)) | ||||||
| 
 | 
 | ||||||
| #define newSlice(T, data, len) T ## _SliceConstruct(data, len) | #define newSlice(T, data, len) T##_SliceConstruct(data, len) | ||||||
| 
 | 
 | ||||||
| arrayTemplate(uint8_t); | arrayTemplate(uint8_t); | ||||||
| sliceTemplate(uint8_t); | sliceTemplate(uint8_t); | ||||||
|  | arrayTemplate(uint32_t); | ||||||
|  | sliceTemplate(uint32_t); | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -1,7 +1,7 @@ | ||||||
| #ifndef ED_FILE_IO_INCLUDED | #ifndef ED_FILE_IO_INCLUDED | ||||||
| #define ED_FILE_IO_INCLUDED | #define ED_FILE_IO_INCLUDED | ||||||
| #include <stdint.h> |  | ||||||
| #include <stdbool.h> | #include <stdbool.h> | ||||||
|  | #include <stdint.h> | ||||||
| 
 | 
 | ||||||
| #include "string.h" | #include "string.h" | ||||||
| 
 | 
 | ||||||
|  | @ -11,12 +11,16 @@ bool load_file(string file_path, size_t size, void *buffer); | ||||||
| #ifdef ED_FILE_IO_IMPLEMENTATION | #ifdef ED_FILE_IO_IMPLEMENTATION | ||||||
| 
 | 
 | ||||||
| #if defined(__unix__) | #if defined(__unix__) | ||||||
|  | #include <assert.h> | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <stdlib.h> | #include <stdlib.h> | ||||||
| #include <assert.h> |  | ||||||
| 
 | 
 | ||||||
| unsigned long get_file_size(const char *filePath) { | unsigned long get_file_size(string file_path) { | ||||||
|     FILE *file = fopen(filePath, "r"); |     char *file_path_with_sentinel = malloc(file_path.len + 1); | ||||||
|  |     memcpy(file_path_with_sentinel, file_path.data, file_path.len + 1); | ||||||
|  |     file_path_with_sentinel[file_path.len] = 0; | ||||||
|  | 
 | ||||||
|  |     FILE *file = fopen(file_path_with_sentinel, "r"); | ||||||
|     assert(file && "get_file_size: failed to open file"); |     assert(file && "get_file_size: failed to open file"); | ||||||
| 
 | 
 | ||||||
|     fseek(file, 0, SEEK_END); |     fseek(file, 0, SEEK_END); | ||||||
|  | @ -25,21 +29,22 @@ unsigned long get_file_size(const char *filePath) { | ||||||
| 
 | 
 | ||||||
|     fclose(file); |     fclose(file); | ||||||
| 
 | 
 | ||||||
|  |     free(file_path_with_sentinel); | ||||||
|     return fsize; |     return fsize; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool load_file(const char *filePath, unsigned long size, void *buffer) { | bool load_file(string file_path, size_t size, void *buffer) { | ||||||
|     FILE *file = fopen(filePath, "r"); |     char *file_path_with_sentinel = malloc(file_path.len + 1); | ||||||
|  |     memcpy(file_path_with_sentinel, file_path.data, file_path.len + 1); | ||||||
|  |     file_path_with_sentinel[file_path.len] = 0; | ||||||
|  | 
 | ||||||
|  |     FILE *file = fopen(file_path_with_sentinel, "r"); | ||||||
|     assert(file && "load_file: failed to open file"); |     assert(file && "load_file: failed to open file"); | ||||||
| 
 | 
 | ||||||
|     fseek(file, 0, SEEK_END); |     fread(buffer, size, 1, file); | ||||||
|     long fsize = ftell(file); |  | ||||||
|     fseek(file, 0, SEEK_SET); |  | ||||||
| 
 |  | ||||||
|     fread(buffer, fsize, 1, file); |  | ||||||
| 
 |  | ||||||
|     fclose(file); |     fclose(file); | ||||||
| 
 | 
 | ||||||
|  |     free(file_path_with_sentinel); | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -47,49 +52,41 @@ bool load_file(const char *filePath, unsigned long size, void *buffer) { | ||||||
| #include <windows.h> | #include <windows.h> | ||||||
| #define assertm(exp, msg) assert(((void)msg, exp)) | #define assertm(exp, msg) assert(((void)msg, exp)) | ||||||
| 
 | 
 | ||||||
| VOID CALLBACK FileIOCompletionRoutine( | VOID CALLBACK FileIOCompletionRoutine(__in DWORD dwErrorCode, | ||||||
|         __in  DWORD dwErrorCode, |                                       __in DWORD dwNumberOfBytesTransfered, | ||||||
|         __in  DWORD dwNumberOfBytesTransfered, |                                       __in LPOVERLAPPED lpOverlapped) { | ||||||
|         __in  LPOVERLAPPED lpOverlapped ) |  | ||||||
| { |  | ||||||
|     printf("Error code: %li", dwErrorCode); |     printf("Error code: %li", dwErrorCode); | ||||||
|     printf("Number of bytes: %li", dwNumberOfBytesTransfered); |     printf("Number of bytes: %li", dwNumberOfBytesTransfered); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| unsigned long get_file_size(const char *filePath) { | unsigned long get_file_size(const char *filePath) { | ||||||
|     HANDLE hFile = CreateFile(filePath, |     HANDLE hFile = | ||||||
|             GENERIC_READ, |         CreateFile(filePath, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, | ||||||
|             FILE_SHARE_READ, |                    FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, NULL); | ||||||
|             NULL, |  | ||||||
|             OPEN_EXISTING, |  | ||||||
|             FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, |  | ||||||
|             NULL); |  | ||||||
| 
 | 
 | ||||||
|     LARGE_INTEGER size; |     LARGE_INTEGER size; | ||||||
|     assert(GetFileSizeEx(hFile, &size) && "get_file_size: failed to get file size"); |     assert(GetFileSizeEx(hFile, &size) && | ||||||
|  |            "get_file_size: failed to get file size"); | ||||||
| 
 | 
 | ||||||
|     return size.LowPart; |     return size.LowPart; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool load_file(const char *filePath, unsigned long size, void *buffer) { | bool load_file(const char *filePath, unsigned long size, void *buffer) { | ||||||
|     HANDLE hFile = CreateFile(filePath, |     HANDLE hFile = CreateFile(filePath, GENERIC_READ, FILE_SHARE_READ, NULL, | ||||||
|             GENERIC_READ, |                               OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL); | ||||||
|             FILE_SHARE_READ, |  | ||||||
|             NULL, |  | ||||||
|             OPEN_EXISTING, |  | ||||||
|             FILE_ATTRIBUTE_NORMAL, |  | ||||||
|             NULL); |  | ||||||
| 
 | 
 | ||||||
|     OVERLAPPED ol = {0}; |     OVERLAPPED ol = {0}; | ||||||
|     unsigned long bytesRead = 0; |     unsigned long bytesRead = 0; | ||||||
|     if (!ReadFile(hFile, buffer, size, &bytesRead, NULL)) { |     if (!ReadFile(hFile, buffer, size, &bytesRead, NULL)) { | ||||||
|         unsigned long error = GetLastError(); |         unsigned long error = GetLastError(); | ||||||
|         unsigned long msg = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, error, 0, buffer, size, NULL); |         unsigned long msg = FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, | ||||||
|  |                                           error, 0, buffer, size, NULL); | ||||||
| 
 | 
 | ||||||
|         printf("failed to load file: %s\n", (char *)buffer); |         printf("failed to load file: %s\n", (char *)buffer); | ||||||
|         assert(false && "failed to read file"); |         assert(false && "failed to read file"); | ||||||
|     } |     } | ||||||
|     assert(bytesRead == size && "load_file: didn't one-shot read the whole file"); |     assert(bytesRead == size && | ||||||
|  |            "load_file: didn't one-shot read the whole file"); | ||||||
| 
 | 
 | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
|  | @ -97,11 +94,14 @@ bool load_file(const char *filePath, unsigned long size, void *buffer) { | ||||||
| #include <Foundation/Foundation.h> | #include <Foundation/Foundation.h> | ||||||
| 
 | 
 | ||||||
| uint64_t get_file_size(string file_path) { | uint64_t get_file_size(string file_path) { | ||||||
|     NSFileManager* file_manager = [NSFileManager defaultManager]; |     NSFileManager *file_manager = [NSFileManager defaultManager]; | ||||||
|     NSString *path = [file_manager stringWithFileSystemRepresentation:(const char *)file_path.data length:file_path.len];  |     NSString *path = [file_manager | ||||||
|  |         stringWithFileSystemRepresentation:(const char *)file_path.data | ||||||
|  |                                     length:file_path.len]; | ||||||
| 
 | 
 | ||||||
|     NSError *error = NULL; |     NSError *error = NULL; | ||||||
|     NSDictionary<NSFileAttributeKey, id> *attributes = [file_manager attributesOfItemAtPath:path error:&error]; |     NSDictionary<NSFileAttributeKey, id> *attributes = | ||||||
|  |         [file_manager attributesOfItemAtPath:path error:&error]; | ||||||
| 
 | 
 | ||||||
|     if (error) { |     if (error) { | ||||||
|         NSLog(@"error getting file attributes: %@\n", error.description); |         NSLog(@"error getting file attributes: %@\n", error.description); | ||||||
|  | @ -109,22 +109,26 @@ uint64_t get_file_size(string file_path) { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     NSNumber *file_size = [attributes objectForKey:NSFileSize]; |     NSNumber *file_size = [attributes objectForKey:NSFileSize]; | ||||||
|      | 
 | ||||||
|     return file_size.unsignedLongLongValue; |     return file_size.unsignedLongLongValue; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool load_file(string file_path, size_t size, void *buffer) { | bool load_file(string file_path, size_t size, void *buffer) { | ||||||
|     NSFileManager* file_manager = [NSFileManager defaultManager]; |     NSFileManager *file_manager = [NSFileManager defaultManager]; | ||||||
|     NSString *path = [file_manager stringWithFileSystemRepresentation:(const char *)file_path.data length:file_path.len];  |     NSString *path = [file_manager | ||||||
|  |         stringWithFileSystemRepresentation:(const char *)file_path.data | ||||||
|  |                                     length:file_path.len]; | ||||||
| 
 | 
 | ||||||
|     NSData *data = [file_manager contentsAtPath:path]; |     NSData *data = [file_manager contentsAtPath:path]; | ||||||
|     if (data == NULL) { |     if (data == NULL) { | ||||||
|         fprintf(stderr, "failed to load file: %.*s\n", (int)file_path.len, file_path.data); |         fprintf(stderr, "failed to load file: %.*s\n", (int)file_path.len, | ||||||
|  |                 file_path.data); | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (data.length > size) { |     if (data.length > size) { | ||||||
|         fprintf(stderr, "buffer not large enough for file: %.*s\n", (int)file_path.len, file_path.data); |         fprintf(stderr, "buffer not large enough for file: %.*s\n", | ||||||
|  |                 (int)file_path.len, file_path.data); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     memcpy(buffer, data.bytes, data.length); |     memcpy(buffer, data.bytes, data.length); | ||||||
|  |  | ||||||
							
								
								
									
										610
									
								
								src/gfx.h
								
								
								
								
							
							
						
						
									
										610
									
								
								src/gfx.h
								
								
								
								
							|  | @ -2,12 +2,6 @@ | ||||||
| 
 | 
 | ||||||
| #ifndef ED_GFX_INCLUDED | #ifndef ED_GFX_INCLUDED | ||||||
| #define ED_GFX_INCLUDED | #define ED_GFX_INCLUDED | ||||||
| #include <AppKit/AppKit.h> |  | ||||||
| #include <CoreGraphics/CoreGraphics.h> |  | ||||||
| #include <Foundation/Foundation.h> |  | ||||||
| #include <Metal/Metal.h> |  | ||||||
| #include <QuartzCore/CoreAnimation.h> |  | ||||||
| #include <QuartzCore/QuartzCore.h> |  | ||||||
| #include <stdint.h> | #include <stdint.h> | ||||||
| 
 | 
 | ||||||
| #include "ed_array.h" | #include "ed_array.h" | ||||||
|  | @ -15,6 +9,13 @@ | ||||||
| 
 | 
 | ||||||
| bool keep_running = true; | bool keep_running = true; | ||||||
| 
 | 
 | ||||||
|  | #if defined(__APPLE__) | ||||||
|  | #include <AppKit/AppKit.h> | ||||||
|  | #include <CoreGraphics/CoreGraphics.h> | ||||||
|  | #include <Foundation/Foundation.h> | ||||||
|  | #include <Metal/Metal.h> | ||||||
|  | #include <QuartzCore/CoreAnimation.h> | ||||||
|  | #include <QuartzCore/QuartzCore.h> | ||||||
| @interface EDGFXView : NSView | @interface EDGFXView : NSView | ||||||
| @property NSTrackingArea *tracking_area; | @property NSTrackingArea *tracking_area; | ||||||
| @end | @end | ||||||
|  | @ -30,6 +31,7 @@ bool keep_running = true; | ||||||
| wrapIdArray(MTLRenderPipelineState); | wrapIdArray(MTLRenderPipelineState); | ||||||
| wrapIdArray(MTLBuffer); | wrapIdArray(MTLBuffer); | ||||||
| wrapIdArray(MTLTexture); | wrapIdArray(MTLTexture); | ||||||
|  | #endif | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     float position[4]; |     float position[4]; | ||||||
|  | @ -50,6 +52,7 @@ arrayTemplate(GpuGlyph); | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     float screen_size[2]; |     float screen_size[2]; | ||||||
|  |     float font_size[2]; | ||||||
| } GpuUniformParams; | } GpuUniformParams; | ||||||
| 
 | 
 | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|  | @ -72,6 +75,63 @@ typedef struct { | ||||||
|     array(_MTLBuffer) buffers; |     array(_MTLBuffer) buffers; | ||||||
|     array(_MTLTexture) textures; |     array(_MTLTexture) textures; | ||||||
| } _metal_gfx_context; | } _metal_gfx_context; | ||||||
|  | #elif __linux__ | ||||||
|  | #include "wayland-crap/xdg-shell.h" | ||||||
|  | #include <EGL/egl.h> | ||||||
|  | #include <GL/gl.h> | ||||||
|  | #include <X11/Xlib.h> | ||||||
|  | #include <sys/mman.h> | ||||||
|  | #include <syscall.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <wayland-client.h> | ||||||
|  | #include <wayland-egl.h> | ||||||
|  | 
 | ||||||
|  | #include "file_io.h" | ||||||
|  | 
 | ||||||
|  | // And I thought MacOS needed a lot of state to create a window
 | ||||||
|  | arrayTemplate(GLuint); | ||||||
|  | typedef struct { | ||||||
|  |     struct wl_display *display; | ||||||
|  |     struct wl_registry *registry; | ||||||
|  |     struct wl_surface *surface; | ||||||
|  |     struct wl_compositor *compositor; | ||||||
|  |     struct wl_shm *shared_memory; | ||||||
|  |     struct wl_shm_pool *shared_memory_pool; | ||||||
|  |     struct xdg_wm_base *wm_base; | ||||||
|  |     struct wl_buffer *buffer; | ||||||
|  |     struct xdg_surface *xdg_surface; | ||||||
|  |     struct xdg_toplevel *xdg_toplevel; | ||||||
|  |     struct wl_seat *seat; | ||||||
|  |     struct wl_pointer *pointer; | ||||||
|  |     uint8_t *pixels; | ||||||
|  | 
 | ||||||
|  |     EGLDisplay egl_display; | ||||||
|  |     EGLConfig egl_config; | ||||||
|  |     EGLSurface egl_surface; | ||||||
|  |     EGLContext egl_context; | ||||||
|  |     struct wl_egl_window *egl_window; | ||||||
|  | 
 | ||||||
|  |     int mouse_x, mouse_y; | ||||||
|  |     int mouse_left_down, mouse_right_down; | ||||||
|  | 
 | ||||||
|  |     GLuint ui_rect_vertex_shader; | ||||||
|  |     GLuint ui_rect_fragment_shader; | ||||||
|  |     GLuint text_atlas_vertex_shader; | ||||||
|  |     GLuint text_atlas_fragment_shader; | ||||||
|  |     GLuint ui_rect_shader_program; | ||||||
|  |     GLuint text_atlas_shader_program; | ||||||
|  | 
 | ||||||
|  |     array(GLuint) buffers; | ||||||
|  |     array(GLuint) textures; | ||||||
|  | } _opengl_gfx_context_wayland; | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_present_wayland(_opengl_gfx_context_wayland *cx); | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     Display *display; | ||||||
|  |     Window window; | ||||||
|  |     int screen; | ||||||
|  | } _opengl_gfx_context_x11; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| typedef void (*_gfx_frame_func)(int mouse_x, int mouse_y, bool mouse_left_down, | typedef void (*_gfx_frame_func)(int mouse_x, int mouse_y, bool mouse_left_down, | ||||||
|  | @ -79,6 +139,9 @@ typedef void (*_gfx_frame_func)(int mouse_x, int mouse_y, bool mouse_left_down, | ||||||
| typedef struct { | typedef struct { | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     _metal_gfx_context backend; |     _metal_gfx_context backend; | ||||||
|  | #elif __linux__ | ||||||
|  |     // TODO: be able to use X11 or Wayland at runtime
 | ||||||
|  |     _opengl_gfx_context_wayland backend; | ||||||
| #else | #else | ||||||
| #error "Unsupported platform" | #error "Unsupported platform" | ||||||
| #endif | #endif | ||||||
|  | @ -205,8 +268,8 @@ void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data, | ||||||
| 
 | 
 | ||||||
| - (void)mouseMoved:(NSEvent *)event { | - (void)mouseMoved:(NSEvent *)event { | ||||||
|     // NSPoint location = NSEvent.mouseLocation;
 |     // NSPoint location = NSEvent.mouseLocation;
 | ||||||
|     NSPoint location = [self convertPoint:[event locationInWindow] |     NSPoint location = | ||||||
|                                  fromView:nil]; |         [self convertPoint:[event locationInWindow] fromView:nil]; | ||||||
| 
 | 
 | ||||||
|     _gfx_context.backend.mouse_x = location.x; |     _gfx_context.backend.mouse_x = location.x; | ||||||
|     _gfx_context.backend.mouse_y = location.y; |     _gfx_context.backend.mouse_y = location.y; | ||||||
|  | @ -302,8 +365,8 @@ static _metal_gfx_context _metal_gfx_init_context(uint32_t width, | ||||||
|         exit(1); |         exit(1); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     id<MTLLibrary> library = [device newLibraryWithURL:libraryURL |     id<MTLLibrary> library = | ||||||
|                                                  error:&libraryError]; |         [device newLibraryWithURL:libraryURL error:&libraryError]; | ||||||
| 
 | 
 | ||||||
|     if (library == NULL) { |     if (library == NULL) { | ||||||
|         if (libraryError.description != NULL) { |         if (libraryError.description != NULL) { | ||||||
|  | @ -433,6 +496,11 @@ static void _metal_gfx_present(_metal_gfx_context *cx) { | ||||||
|                 (float)_gfx_context.frame_width, |                 (float)_gfx_context.frame_width, | ||||||
|                 (float)_gfx_context.frame_height, |                 (float)_gfx_context.frame_height, | ||||||
|             }, |             }, | ||||||
|  |         .font_size = | ||||||
|  |             { | ||||||
|  |                 (float)_FONT_WIDTH, | ||||||
|  |                 (float)_FONT_HEIGHT, | ||||||
|  |             }, | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, |     gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, | ||||||
|  | @ -573,11 +641,519 @@ static void _metal_gfx_update_buffer(_metal_gfx_context *cx, | ||||||
|     // FIXME: actually check to see if this will fit in the buffer
 |     // FIXME: actually check to see if this will fit in the buffer
 | ||||||
|     memcpy(buffer_contents, data, len); |     memcpy(buffer_contents, data, len); | ||||||
| } | } | ||||||
|  | #elif __linux__ | ||||||
|  | #include <linux/input-event-codes.h> | ||||||
|  | 
 | ||||||
|  | static void _wayland_pointer_enter(void *data, struct wl_pointer *pointer, | ||||||
|  |                                    uint32_t serial, struct wl_surface *surface, | ||||||
|  |                                    wl_fixed_t x, wl_fixed_t y) { | ||||||
|  |     // fprintf(stderr, "pointer enter: %d, %d\n", x, y);
 | ||||||
|  | } | ||||||
|  | static void _wayland_pointer_leave(void *data, struct wl_pointer *pointer, | ||||||
|  |                                    uint32_t serial, | ||||||
|  |                                    struct wl_surface *surface) { | ||||||
|  |     // fprintf(stderr, "pointer leave\n");
 | ||||||
|  | } | ||||||
|  | static void _wayland_pointer_button(void *data, struct wl_pointer *pointer, | ||||||
|  |                                     uint32_t serial, uint32_t time, | ||||||
|  |                                     uint32_t button, uint32_t state) { | ||||||
|  |     _opengl_gfx_context_wayland *cx = data; | ||||||
|  | 
 | ||||||
|  |     if (state == WL_POINTER_BUTTON_STATE_PRESSED) { | ||||||
|  |         if (button == BTN_LEFT) { | ||||||
|  |             cx->mouse_left_down = true; | ||||||
|  |         } else if (button == BTN_RIGHT) { | ||||||
|  |             cx->mouse_right_down = true; | ||||||
|  |         } | ||||||
|  |     } else if (state == WL_POINTER_BUTTON_STATE_RELEASED) { | ||||||
|  |         if (button == BTN_LEFT) { | ||||||
|  |             cx->mouse_left_down = false; | ||||||
|  |         } else if (button == BTN_RIGHT) { | ||||||
|  |             cx->mouse_right_down = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | static void _wayland_pointer_axis(void *data, struct wl_pointer *pointer, | ||||||
|  |                                   uint32_t time, uint32_t axis, | ||||||
|  |                                   wl_fixed_t value) {} | ||||||
|  | static void _wayland_pointer_motion(void *data, struct wl_pointer *pointer, | ||||||
|  |                                     uint32_t time, wl_fixed_t x, wl_fixed_t y) { | ||||||
|  |     _opengl_gfx_context_wayland *cx = data; | ||||||
|  |     cx->mouse_x = wl_fixed_to_int(x); | ||||||
|  |     cx->mouse_y = wl_fixed_to_int(y); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct wl_pointer_listener pointer_listener = { | ||||||
|  |     .enter = _wayland_pointer_enter, | ||||||
|  |     .leave = _wayland_pointer_leave, | ||||||
|  |     .motion = _wayland_pointer_motion, | ||||||
|  |     .button = _wayland_pointer_button, | ||||||
|  |     .axis = _wayland_pointer_axis, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void _wayland_xdg_toplevel_configure(void *data, | ||||||
|  |                                             struct xdg_toplevel *xdg_toplevel, | ||||||
|  |                                             int32_t width, int32_t height, | ||||||
|  |                                             struct wl_array *states) {} | ||||||
|  | 
 | ||||||
|  | static void _wayland_xdg_toplevel_close(void *data, | ||||||
|  |                                         struct xdg_toplevel *xdg_toplevel) { | ||||||
|  |     keep_running = false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct xdg_toplevel_listener xdg_toplevel_listener = { | ||||||
|  |     _wayland_xdg_toplevel_configure, | ||||||
|  |     _wayland_xdg_toplevel_close, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void _wayland_xdg_surface_configure(void *data, | ||||||
|  |                                            struct xdg_surface *xdg_surface, | ||||||
|  |                                            uint32_t serial) { | ||||||
|  |     xdg_surface_ack_configure(xdg_surface, serial); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static const struct xdg_surface_listener xdg_surface_listener = { | ||||||
|  |     _wayland_xdg_surface_configure, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void _wayland_xdg_wm_base_ping(void *data, struct xdg_wm_base *shell, | ||||||
|  |                                       uint32_t serial) { | ||||||
|  |     xdg_wm_base_pong(shell, serial); | ||||||
|  | } | ||||||
|  | static const struct xdg_wm_base_listener xdg_wm_base_listener = { | ||||||
|  |     _wayland_xdg_wm_base_ping, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void _wayland_registry_handle_global(void *data, | ||||||
|  |                                             struct wl_registry *registry, | ||||||
|  |                                             uint32_t name, | ||||||
|  |                                             const char *interface, | ||||||
|  |                                             uint32_t version) { | ||||||
|  |     _opengl_gfx_context_wayland *d = data; | ||||||
|  | 
 | ||||||
|  |     fprintf(stderr, "global: %s\n", interface); | ||||||
|  | 
 | ||||||
|  |     if (strcmp(interface, "wl_compositor") == 0) { | ||||||
|  |         d->compositor = | ||||||
|  |             wl_registry_bind(registry, name, &wl_compositor_interface, 3); | ||||||
|  |     } else if (strcmp(interface, "wl_shm") == 0) { | ||||||
|  |         d->shared_memory = | ||||||
|  |             wl_registry_bind(registry, name, &wl_shm_interface, 1); | ||||||
|  |     } else if (strcmp(interface, "xdg_wm_base") == 0) { | ||||||
|  |         d->wm_base = | ||||||
|  |             wl_registry_bind(registry, name, &xdg_wm_base_interface, 1); | ||||||
|  |     } else if (strcmp(interface, "wl_seat") == 0) { | ||||||
|  |         d->seat = wl_registry_bind(registry, name, &wl_seat_interface, 1); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _wayland_registry_handle_global_remove(void *data, | ||||||
|  |                                                    struct wl_registry *registry, | ||||||
|  |                                                    uint32_t name) {} | ||||||
|  | 
 | ||||||
|  | static const struct wl_registry_listener registry_listener = { | ||||||
|  |     _wayland_registry_handle_global, | ||||||
|  |     _wayland_registry_handle_global_remove, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_message_callback(GLenum source, GLenum type, GLenum id, | ||||||
|  |                                          GLenum severity, GLsizei length, | ||||||
|  |                                          const GLchar *message, | ||||||
|  |                                          const void *user_param) { | ||||||
|  |     fprintf(stderr, | ||||||
|  |             "GL CALLBACK: %s type = 0x%x, severity = 0x%x, message = %s\n", | ||||||
|  |             type == GL_DEBUG_TYPE_ERROR ? "** GL ERROR **" : "", type, severity, | ||||||
|  |             message); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_check_shader_error(string msg, GLuint shader, | ||||||
|  |                                            GLuint status) { | ||||||
|  |     GLint good = 0; | ||||||
|  |     glGetShaderiv(shader, status, &good); | ||||||
|  |     if (good == GL_FALSE) { | ||||||
|  |         GLint max_length = 0; | ||||||
|  |         glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &max_length); | ||||||
|  | 
 | ||||||
|  |         uint8_t *log_buffer = malloc(max_length + 1); | ||||||
|  |         glGetShaderInfoLog(shader, max_length, &max_length, log_buffer); | ||||||
|  |         glDeleteShader(shader); | ||||||
|  | 
 | ||||||
|  |         fprintf(stderr, "%.*s: %.*s\n", msg.len, msg.data, max_length, | ||||||
|  |                 log_buffer); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_check_shader_program_error(string msg, | ||||||
|  |                                                    GLuint shader_program, | ||||||
|  |                                                    GLuint status) { | ||||||
|  |     GLint good = 0; | ||||||
|  |     glGetProgramiv(shader_program, status, &good); | ||||||
|  |     if (good == GL_FALSE) { | ||||||
|  |         GLint max_length = 0; | ||||||
|  |         glGetProgramiv(shader_program, GL_INFO_LOG_LENGTH, &max_length); | ||||||
|  | 
 | ||||||
|  |         uint8_t *log_buffer = malloc(max_length + 1); | ||||||
|  |         glGetProgramInfoLog(shader_program, max_length, &max_length, | ||||||
|  |                             log_buffer); | ||||||
|  |         glDeleteProgram(shader_program); | ||||||
|  | 
 | ||||||
|  |         fprintf(stderr, "%.*s: %.*s\n", msg.len, msg.data, max_length, | ||||||
|  |                 log_buffer); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static GLuint _opengl_gfx_compile_shader(string file_path, GLuint shader_type) { | ||||||
|  |     GLuint shader = glCreateShader(shader_type); | ||||||
|  |     size_t shader_file_size = get_file_size(file_path); | ||||||
|  |     uint8_t *shader_file_data = malloc(shader_file_size + 1); | ||||||
|  |     load_file(file_path, shader_file_size, shader_file_data); | ||||||
|  |     shader_file_data[shader_file_size] = 0; | ||||||
|  |     // fprintf(stderr, "%s\n", shader_file_data);
 | ||||||
|  | 
 | ||||||
|  |     glShaderSource(shader, 1, &shader_file_data, NULL); | ||||||
|  |     glCompileShader(shader); | ||||||
|  | 
 | ||||||
|  |     _opengl_gfx_check_shader_error(_String("failed to compile shader"), shader, | ||||||
|  |                                    GL_COMPILE_STATUS); | ||||||
|  | 
 | ||||||
|  |     return shader; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static _opengl_gfx_context_wayland | ||||||
|  | _opengl_gfx_init_context_wayland(uint32_t width, uint32_t height) { | ||||||
|  |     _opengl_gfx_context_wayland cx = {0}; | ||||||
|  | 
 | ||||||
|  |     cx.display = wl_display_connect(NULL); | ||||||
|  |     if (!cx.display) { | ||||||
|  |         fprintf(stderr, "Failed to connect to Wayland display\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |     struct wl_registry *registry = wl_display_get_registry(cx.display); | ||||||
|  |     wl_registry_add_listener(registry, ®istry_listener, &cx); | ||||||
|  | 
 | ||||||
|  |     // wait for all the globals to be registered
 | ||||||
|  |     wl_display_roundtrip(cx.display); | ||||||
|  |     xdg_wm_base_add_listener(cx.wm_base, &xdg_wm_base_listener, | ||||||
|  |                              &_gfx_context.backend); | ||||||
|  | 
 | ||||||
|  |     cx.pointer = wl_seat_get_pointer(cx.seat); | ||||||
|  |     wl_pointer_add_listener(cx.pointer, &pointer_listener, | ||||||
|  |                             &_gfx_context.backend); | ||||||
|  | 
 | ||||||
|  |     cx.surface = wl_compositor_create_surface(cx.compositor); | ||||||
|  |     cx.xdg_surface = xdg_wm_base_get_xdg_surface(cx.wm_base, cx.surface); | ||||||
|  |     xdg_surface_add_listener(cx.xdg_surface, &xdg_surface_listener, | ||||||
|  |                              &_gfx_context.backend); | ||||||
|  | 
 | ||||||
|  |     cx.xdg_toplevel = xdg_surface_get_toplevel(cx.xdg_surface); | ||||||
|  |     xdg_toplevel_add_listener(cx.xdg_toplevel, &xdg_toplevel_listener, | ||||||
|  |                               &_gfx_context.backend); | ||||||
|  |     xdg_toplevel_set_title(cx.xdg_toplevel, "chat - [Slack sux]"); | ||||||
|  |     xdg_toplevel_set_app_id(cx.xdg_toplevel, "nl.spacegirl.a_chat_client"); | ||||||
|  | 
 | ||||||
|  |     wl_surface_commit(cx.surface); | ||||||
|  |     wl_display_roundtrip(cx.display); | ||||||
|  | 
 | ||||||
|  |     int buffer_size = width * height * 4; | ||||||
|  |     int fd = syscall(SYS_memfd_create, "buffer", 0); | ||||||
|  |     ftruncate(fd, buffer_size); | ||||||
|  | 
 | ||||||
|  |     cx.pixels = | ||||||
|  |         mmap(NULL, buffer_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0); | ||||||
|  |     for (int i = 0; i < buffer_size; i++) { | ||||||
|  |         cx.pixels[i] = 0; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     cx.shared_memory_pool = | ||||||
|  |         wl_shm_create_pool(cx.shared_memory, fd, buffer_size); | ||||||
|  |     cx.buffer = | ||||||
|  |         wl_shm_pool_create_buffer(cx.shared_memory_pool, 0, width, height, | ||||||
|  |                                   width * 4, WL_SHM_FORMAT_ARGB8888); | ||||||
|  | 
 | ||||||
|  |     wl_surface_attach(cx.surface, cx.buffer, 0, 0); | ||||||
|  |     wl_surface_commit(cx.surface); | ||||||
|  | 
 | ||||||
|  |     /* Init EGL */ | ||||||
|  |     EGLint major, minor, count, n, size; | ||||||
|  |     EGLConfig *configs; | ||||||
|  |     EGLint config_attribs[] = { | ||||||
|  |         EGL_SURFACE_TYPE, | ||||||
|  |         EGL_WINDOW_BIT, | ||||||
|  |         //
 | ||||||
|  |         EGL_RED_SIZE, | ||||||
|  |         8, | ||||||
|  |         //
 | ||||||
|  |         EGL_BLUE_SIZE, | ||||||
|  |         8, | ||||||
|  |         //
 | ||||||
|  |         EGL_GREEN_SIZE, | ||||||
|  |         8, | ||||||
|  |         //
 | ||||||
|  |         EGL_RENDERABLE_TYPE, | ||||||
|  |         EGL_OPENGL_BIT, | ||||||
|  |         //
 | ||||||
|  |         EGL_NONE, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     static const EGLint context_attribs[] = {EGL_CONTEXT_CLIENT_VERSION, 2, | ||||||
|  |                                              EGL_NONE}; | ||||||
|  | 
 | ||||||
|  |     cx.egl_display = eglGetDisplay(cx.display); | ||||||
|  |     if (cx.egl_display == EGL_NO_DISPLAY) { | ||||||
|  |         fprintf(stderr, "Failed to create EGL display\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |     fprintf(stderr, "Created EGL display\n"); | ||||||
|  | 
 | ||||||
|  |     if (eglInitialize(cx.egl_display, &major, &minor) != EGL_TRUE) { | ||||||
|  |         fprintf(stderr, "Failed to initialize EGL display\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |     fprintf(stderr, "EGL major: %d, minor: %d\n", major, minor); | ||||||
|  | 
 | ||||||
|  |     eglGetConfigs(cx.egl_display, NULL, 0, &count); | ||||||
|  |     configs = calloc(count, sizeof(EGLConfig)); | ||||||
|  | 
 | ||||||
|  |     eglChooseConfig(cx.egl_display, config_attribs, configs, count, &n); | ||||||
|  | 
 | ||||||
|  |     for (int i = 0; i < n; ++i) { | ||||||
|  |         eglGetConfigAttrib(cx.egl_display, configs[i], EGL_BUFFER_SIZE, &size); | ||||||
|  |         fprintf(stderr, "EGL Buffer size: %d\n", size); | ||||||
|  | 
 | ||||||
|  |         eglGetConfigAttrib(cx.egl_display, configs[i], EGL_RED_SIZE, &size); | ||||||
|  |         fprintf(stderr, "EGL Red size: %d\n", size); | ||||||
|  | 
 | ||||||
|  |         cx.egl_config = configs[i]; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     eglBindAPI(EGL_OPENGL_API); | ||||||
|  |     cx.egl_context = eglCreateContext(cx.egl_display, cx.egl_config, | ||||||
|  |                                       EGL_NO_CONTEXT, context_attribs); | ||||||
|  | 
 | ||||||
|  |     cx.egl_window = wl_egl_window_create(cx.surface, width, height); | ||||||
|  |     if (cx.egl_window == EGL_NO_SURFACE) { | ||||||
|  |         fprintf(stderr, "Failed to create EGL window\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  |     fprintf(stderr, "Created EGL window\n"); | ||||||
|  | 
 | ||||||
|  |     cx.egl_surface = eglCreateWindowSurface(cx.egl_display, cx.egl_config, | ||||||
|  |                                             cx.egl_window, NULL); | ||||||
|  |     if (eglMakeCurrent(cx.egl_display, cx.egl_surface, cx.egl_surface, | ||||||
|  |                        cx.egl_context) != EGL_TRUE) { | ||||||
|  |         fprintf(stderr, "eglMakeCurrent() failed\n"); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     glEnable(GL_DEBUG_OUTPUT); | ||||||
|  |     glDebugMessageCallback(_opengl_gfx_message_callback, NULL); | ||||||
|  | 
 | ||||||
|  |     glEnable(GL_BLEND); | ||||||
|  |     glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); | ||||||
|  |     glBlendFunc(GL_DST_ALPHA, GL_ONE_MINUS_SRC_ALPHA); | ||||||
|  | 
 | ||||||
|  |     cx.ui_rect_vertex_shader = _opengl_gfx_compile_shader( | ||||||
|  |         _String("shaders/ui_rect_vertex.glsl"), GL_VERTEX_SHADER); | ||||||
|  |     cx.ui_rect_fragment_shader = _opengl_gfx_compile_shader( | ||||||
|  |         _String("shaders/ui_rect_fragment.glsl"), GL_FRAGMENT_SHADER); | ||||||
|  |     cx.ui_rect_shader_program = glCreateProgram(); | ||||||
|  |     glAttachShader(cx.ui_rect_shader_program, cx.ui_rect_vertex_shader); | ||||||
|  |     glAttachShader(cx.ui_rect_shader_program, cx.ui_rect_fragment_shader); | ||||||
|  |     glLinkProgram(cx.ui_rect_shader_program); | ||||||
|  |     _opengl_gfx_check_shader_program_error( | ||||||
|  |         _String("failed to link ui_rect shader program"), | ||||||
|  |         cx.ui_rect_shader_program, GL_LINK_STATUS); | ||||||
|  | 
 | ||||||
|  |     cx.text_atlas_vertex_shader = _opengl_gfx_compile_shader( | ||||||
|  |         _String("shaders/text_atlas_vertex.glsl"), GL_VERTEX_SHADER); | ||||||
|  |     cx.text_atlas_fragment_shader = _opengl_gfx_compile_shader( | ||||||
|  |         _String("shaders/text_atlas_fragment.glsl"), GL_FRAGMENT_SHADER); | ||||||
|  |     cx.text_atlas_shader_program = glCreateProgram(); | ||||||
|  |     glAttachShader(cx.text_atlas_shader_program, cx.text_atlas_vertex_shader); | ||||||
|  |     glAttachShader(cx.text_atlas_shader_program, cx.text_atlas_fragment_shader); | ||||||
|  |     glLinkProgram(cx.text_atlas_shader_program); | ||||||
|  |     _opengl_gfx_check_shader_program_error( | ||||||
|  |         _String("failed to link text_atlas shader program"), | ||||||
|  |         cx.text_atlas_shader_program, GL_LINK_STATUS); | ||||||
|  | 
 | ||||||
|  |     cx.buffers = newArray(GLuint, 8); | ||||||
|  |     cx.textures = newArray(GLuint, 8); | ||||||
|  |     /* ******** */ | ||||||
|  | 
 | ||||||
|  |     return cx; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static _opengl_gfx_context_x11 _opengl_gfx_init_context_11(uint32_t width, | ||||||
|  |                                                            uint32_t height) { | ||||||
|  |     Display *display = XOpenDisplay(NULL); | ||||||
|  |     if (display == NULL) { | ||||||
|  |         fprintf(stderr, "Failed to open X display\n"); | ||||||
|  |         exit(1); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     int screen = DefaultScreen(display); | ||||||
|  |     Window window = XCreateSimpleWindow( | ||||||
|  |         display, RootWindow(display, screen), 0, 0, width, height, 1, | ||||||
|  |         BlackPixel(display, screen), WhitePixel(display, screen)); | ||||||
|  |     XSelectInput(display, window, | ||||||
|  |                  ExposureMask | KeyPressMask | ButtonPressMask | | ||||||
|  |                      ButtonReleaseMask); | ||||||
|  |     XMapWindow(display, window); | ||||||
|  | 
 | ||||||
|  |     return (_opengl_gfx_context_x11){ | ||||||
|  |         .display = display, | ||||||
|  |         .window = window, | ||||||
|  |         .screen = screen, | ||||||
|  |     }; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_send_events_wayland(_opengl_gfx_context_wayland *cx) { | ||||||
|  |     wl_display_dispatch(cx->display); | ||||||
|  | 
 | ||||||
|  |     // TODO: don't just render like crazy, limit framerate
 | ||||||
|  |     _opengl_gfx_present_wayland(cx); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_send_events_x11(_opengl_gfx_context_x11 *cx) { | ||||||
|  |     XEvent e; | ||||||
|  |     XNextEvent(cx->display, &e); | ||||||
|  |     if (e.type == Expose) { | ||||||
|  |         XFillRectangle(cx->display, cx->window, | ||||||
|  |                        DefaultGC(cx->display, cx->screen), 20, 20, 10, 10); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (e.type == KeyPress) { | ||||||
|  |         keep_running = false; | ||||||
|  | 
 | ||||||
|  |         XCloseDisplay(cx->display); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void _opengl_gfx_present_wayland(_opengl_gfx_context_wayland *cx) { | ||||||
|  |     _gfx_context.gpu_glyphs.size = 0; | ||||||
|  |     _gfx_context.gpu_ui_rects.size = 0; | ||||||
|  |     _gfx_context.frame_func(cx->mouse_x, cx->mouse_y, cx->mouse_left_down, | ||||||
|  |                             cx->mouse_right_down); | ||||||
|  | 
 | ||||||
|  |     if (_gfx_context.gpu_glyphs.size > 0) { | ||||||
|  |         gfx_update_buffer(&_gfx_context, 2, _gfx_context.gpu_glyphs.data, | ||||||
|  |                           _gfx_context.gpu_glyphs.size * sizeof(GpuGlyph)); | ||||||
|  |     } | ||||||
|  |     if (_gfx_context.gpu_ui_rects.size > 0) { | ||||||
|  |         gfx_update_buffer(&_gfx_context, 4, _gfx_context.gpu_ui_rects.data, | ||||||
|  |                           _gfx_context.gpu_ui_rects.size * sizeof(GpuUiRect)); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     GpuUniformParams gpu_uniform_params = { | ||||||
|  |         .screen_size = | ||||||
|  |             { | ||||||
|  |                 (float)_gfx_context.frame_width, | ||||||
|  |                 (float)_gfx_context.frame_height, | ||||||
|  |             }, | ||||||
|  |         .font_size = | ||||||
|  |             { | ||||||
|  |                 (float)_FONT_WIDTH, | ||||||
|  |                 (float)_FONT_HEIGHT, | ||||||
|  |             }, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     gfx_update_buffer(&_gfx_context, 3, &gpu_uniform_params, | ||||||
|  |                       sizeof(GpuUniformParams)); | ||||||
|  |     glClearColor(0.0f, 0.0f, 0.0f, 1.0f); | ||||||
|  |     glClear(GL_COLOR_BUFFER_BIT); | ||||||
|  |     glViewport(0, 0, _gfx_context.frame_width, _gfx_context.frame_height); | ||||||
|  | 
 | ||||||
|  |     if (_gfx_context.gpu_ui_rects.size > 0) { | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, cx->buffers.data[0]); | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 1, cx->buffers.data[4]); | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 2, cx->buffers.data[3]); | ||||||
|  |         glUseProgram(cx->ui_rect_shader_program); | ||||||
|  | 
 | ||||||
|  |         const uint16_t indices[] = {0, 1, 2, 0, 2, 3}; | ||||||
|  |         glDrawElementsInstanced(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, indices, | ||||||
|  |                                 _gfx_context.gpu_ui_rects.size); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (_gfx_context.gpu_glyphs.size > 0) { | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, cx->buffers.data[0]); | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 1, cx->buffers.data[2]); | ||||||
|  |         glBindBufferBase(GL_SHADER_STORAGE_BUFFER, 2, cx->buffers.data[3]); | ||||||
|  |         glUseProgram(cx->text_atlas_shader_program); | ||||||
|  | 
 | ||||||
|  |         const uint16_t indices[] = {0, 1, 2, 0, 2, 3}; | ||||||
|  |         glDrawElementsInstanced(GL_TRIANGLES, 6, GL_UNSIGNED_SHORT, indices, | ||||||
|  |                                 _gfx_context.gpu_glyphs.size); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     glFlush(); | ||||||
|  | 
 | ||||||
|  |     if (eglSwapBuffers(cx->egl_display, cx->egl_surface) != EGL_TRUE) { | ||||||
|  |         fprintf(stderr, "eglSwapBuffers() failed\n"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static size_t | ||||||
|  | _opengl_gfx_push_texture_buffer_wayland(_opengl_gfx_context_wayland *cx, | ||||||
|  |                                         uint32_t width, uint32_t height, | ||||||
|  |                                         const void *data, size_t len) { | ||||||
|  |     pushArray(GLuint, &cx->textures, 0); | ||||||
|  |     glCreateTextures(GL_TEXTURE_2D, 1, | ||||||
|  |                      &cx->textures.data[cx->textures.size - 1]); | ||||||
|  |     glTextureParameteri(cx->textures.data[cx->textures.size - 1], | ||||||
|  |                         GL_TEXTURE_MIN_FILTER, GL_NEAREST); | ||||||
|  |     glTextureParameteri(cx->textures.data[cx->textures.size - 1], | ||||||
|  |                         GL_TEXTURE_MAG_FILTER, GL_NEAREST); | ||||||
|  |     glTextureStorage2D(cx->textures.data[cx->textures.size - 1], 1, GL_RGBA8, | ||||||
|  |                        width, height); | ||||||
|  |     glTextureSubImage2D(cx->textures.data[cx->textures.size - 1], 0, 0, 0, | ||||||
|  |                         width, height, GL_RED, GL_UNSIGNED_BYTE, data); | ||||||
|  |     glBindTextureUnit(cx->textures.size - 1, | ||||||
|  |                       cx->textures.data[cx->textures.size - 1]); | ||||||
|  | 
 | ||||||
|  |     return cx->textures.size - 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | _opengl_gfx_resize_texture_buffer_wayland(_opengl_gfx_context_wayland *cx, | ||||||
|  |                                           uint32_t width, size_t texture_index, | ||||||
|  |                                           uint32_t height) { | ||||||
|  |     // TODO
 | ||||||
|  |     assert(false && "_opengl_gfx_resize_texture_buffer_wayland unimplemented"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | size_t _opengl_gfx_push_vertex_buffer_wayland(_opengl_gfx_context_wayland *cx, | ||||||
|  |                                               const void *data, size_t len) { | ||||||
|  |     pushArray(GLuint, &cx->buffers, 0); | ||||||
|  |     glCreateBuffers(1, &cx->buffers.data[cx->buffers.size - 1]); | ||||||
|  |     glNamedBufferStorage(cx->buffers.data[cx->buffers.size - 1], len, data, | ||||||
|  |                          GL_DYNAMIC_STORAGE_BIT); | ||||||
|  | 
 | ||||||
|  |     return cx->buffers.size - 1; | ||||||
|  | } | ||||||
|  | static size_t | ||||||
|  | _opengl_gfx_allocate_vertex_buffer_wayland(_opengl_gfx_context_wayland *cx, | ||||||
|  |                                            size_t len) { | ||||||
|  |     pushArray(GLuint, &cx->buffers, 0); | ||||||
|  |     glCreateBuffers(1, &cx->buffers.data[cx->buffers.size - 1]); | ||||||
|  |     glNamedBufferStorage(cx->buffers.data[cx->buffers.size - 1], len, (void *)0, | ||||||
|  |                          GL_DYNAMIC_STORAGE_BIT); | ||||||
|  | 
 | ||||||
|  |     return cx->buffers.size - 1; | ||||||
|  | } | ||||||
|  | static void _opengl_gfx_update_buffer_wayland(_opengl_gfx_context_wayland *cx, | ||||||
|  |                                               size_t buffer_index, | ||||||
|  |                                               const void *data, size_t len) { | ||||||
|  |     glNamedBufferSubData(cx->buffers.data[buffer_index], 0, len, data); | ||||||
|  | } | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| void gfx_run_events(gfx_context_t *cx) { | void gfx_run_events(gfx_context_t *cx) { | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     return _metal_gfx_send_events(&cx->backend); |     return _metal_gfx_send_events(&cx->backend); | ||||||
|  | #elif __linux__ | ||||||
|  |     return _opengl_gfx_send_events_wayland(&cx->backend); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -588,6 +1164,9 @@ size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width, | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     return _metal_gfx_push_texture_buffer(&cx->backend, width, height, data, |     return _metal_gfx_push_texture_buffer(&cx->backend, width, height, data, | ||||||
|                                           len); |                                           len); | ||||||
|  | #elif __linux__ | ||||||
|  |     return _opengl_gfx_push_texture_buffer_wayland(&cx->backend, width, height, | ||||||
|  |                                                    data, len); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -596,6 +1175,8 @@ size_t gfx_push_texture_buffer(gfx_context_t *cx, uint32_t width, | ||||||
| size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { | size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     return _metal_gfx_push_vertex_buffer(&cx->backend, data, len); |     return _metal_gfx_push_vertex_buffer(&cx->backend, data, len); | ||||||
|  | #elif __linux__ | ||||||
|  |     return _opengl_gfx_push_vertex_buffer_wayland(&cx->backend, data, len); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -604,6 +1185,8 @@ size_t gfx_push_vertex_buffer(gfx_context_t *cx, const void *data, size_t len) { | ||||||
| size_t gfx_allocate_vertex_buffer(gfx_context_t *cx, size_t len) { | size_t gfx_allocate_vertex_buffer(gfx_context_t *cx, size_t len) { | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     return _metal_gfx_allocate_vertex_buffer(&cx->backend, len); |     return _metal_gfx_allocate_vertex_buffer(&cx->backend, len); | ||||||
|  | #elif __linux__ | ||||||
|  |     return _opengl_gfx_allocate_vertex_buffer_wayland(&cx->backend, len); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -614,6 +1197,9 @@ void gfx_update_buffer(gfx_context_t *cx, size_t buffer_index, const void *data, | ||||||
|                        size_t len) { |                        size_t len) { | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|     return _metal_gfx_update_buffer(&cx->backend, buffer_index, data, len); |     return _metal_gfx_update_buffer(&cx->backend, buffer_index, data, len); | ||||||
|  | #elif __linux__ | ||||||
|  |     return _opengl_gfx_update_buffer_wayland(&cx->backend, buffer_index, data, | ||||||
|  |                                              len); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -661,6 +1247,8 @@ void *gfx_init_context(_gfx_frame_func frame_func, uint32_t width, | ||||||
|     __auto_type backend = |     __auto_type backend = | ||||||
| #if defined(__APPLE__) | #if defined(__APPLE__) | ||||||
|         _metal_gfx_init_context(width, height); |         _metal_gfx_init_context(width, height); | ||||||
|  | #elif __linux__ | ||||||
|  |         _opengl_gfx_init_context_wayland(width, height); | ||||||
| #else | #else | ||||||
| #error "Unsupported graphics backend" | #error "Unsupported graphics backend" | ||||||
| #endif | #endif | ||||||
|  | @ -674,7 +1262,7 @@ void *gfx_init_context(_gfx_frame_func frame_func, uint32_t width, | ||||||
|     _gfx_context.gpu_glyphs = newArray(GpuGlyph, 8192 * 32); |     _gfx_context.gpu_glyphs = newArray(GpuGlyph, 8192 * 32); | ||||||
|     _gfx_context.glyph_cache = newArray(GpuGlyph, 97); |     _gfx_context.glyph_cache = newArray(GpuGlyph, 97); | ||||||
| 
 | 
 | ||||||
|     float vertices[] = { |     const float vertices[] = { | ||||||
|         // positions       texture coords
 |         // positions       texture coords
 | ||||||
|         -1.0f, 1.0f,  0.0f, 0.0f, 1.0f,  1.0f,  1.0f, 0.0f, |         -1.0f, 1.0f,  0.0f, 0.0f, 1.0f,  1.0f,  1.0f, 0.0f, | ||||||
|         1.0f,  -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 0.0f, 1.0f, |         1.0f,  -1.0f, 1.0f, 1.0f, -1.0f, -1.0f, 0.0f, 1.0f, | ||||||
|  |  | ||||||
							
								
								
									
										28
									
								
								src/ht.h
								
								
								
								
							
							
						
						
									
										28
									
								
								src/ht.h
								
								
								
								
							|  | @ -3,10 +3,10 @@ | ||||||
| #ifndef ED_HT_INCLUDED | #ifndef ED_HT_INCLUDED | ||||||
| #define ED_HT_INCLUDED | #define ED_HT_INCLUDED | ||||||
| 
 | 
 | ||||||
| #include <stdlib.h> |  | ||||||
| #include <stddef.h> |  | ||||||
| #include <stdbool.h> | #include <stdbool.h> | ||||||
|  | #include <stddef.h> | ||||||
| #include <stdint.h> | #include <stdint.h> | ||||||
|  | #include <stdlib.h> | ||||||
| #include <string.h> | #include <string.h> | ||||||
| 
 | 
 | ||||||
| #include "string.h" | #include "string.h" | ||||||
|  | @ -33,7 +33,7 @@ typedef uint64_t ht_hash_t; | ||||||
| static ht_hash_t ht_hash(string key, uint64_t mod) { | static ht_hash_t ht_hash(string key, uint64_t mod) { | ||||||
|     ht_hash_t hash = FNV_OFFSET; |     ht_hash_t hash = FNV_OFFSET; | ||||||
| 
 | 
 | ||||||
|     for (size_t i=0; i<key.len; ++i) { |     for (size_t i = 0; i < key.len; ++i) { | ||||||
|         hash *= FNV_PRIME; |         hash *= FNV_PRIME; | ||||||
|         hash ^= key.data[i]; |         hash ^= key.data[i]; | ||||||
|     } |     } | ||||||
|  | @ -42,13 +42,14 @@ static ht_hash_t ht_hash(string key, uint64_t mod) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| ed_ht ht_create(size_t max_entries, size_t value_size) { | ed_ht ht_create(size_t max_entries, size_t value_size) { | ||||||
|     ed_ht_slot *key_slots = (ed_ht_slot *)malloc(max_entries * sizeof(ed_ht_slot)); |     ed_ht_slot *key_slots = | ||||||
|  |         (ed_ht_slot *)malloc(max_entries * sizeof(ed_ht_slot)); | ||||||
|     void *value_slots = malloc(max_entries * value_size); |     void *value_slots = malloc(max_entries * value_size); | ||||||
| 
 | 
 | ||||||
|     memset(key_slots, 0, max_entries * sizeof(ed_ht_slot)); |     memset(key_slots, 0, max_entries * sizeof(ed_ht_slot)); | ||||||
|     memset(value_slots, 0, max_entries * value_size); |     memset(value_slots, 0, max_entries * value_size); | ||||||
| 
 | 
 | ||||||
|     return (ed_ht) { |     return (ed_ht){ | ||||||
|         .key_slots = key_slots, |         .key_slots = key_slots, | ||||||
|         .value_slots = value_slots, |         .value_slots = value_slots, | ||||||
|         .value_size = value_size, |         .value_size = value_size, | ||||||
|  | @ -59,14 +60,15 @@ ed_ht ht_create(size_t max_entries, size_t value_size) { | ||||||
| bool ht_set(ed_ht *ht, string key, void *value) { | bool ht_set(ed_ht *ht, string key, void *value) { | ||||||
|     ht_hash_t hash = ht_hash(key, ht->capacity); |     ht_hash_t hash = ht_hash(key, ht->capacity); | ||||||
| 
 | 
 | ||||||
|     for (size_t i=hash; i<ht->capacity; ++i) { |     for (size_t i = hash; i < ht->capacity; ++i) { | ||||||
|         if (ht->key_slots[i].key.data == NULL || string_eq(ht->key_slots[i].key, key)) { |         if (ht->key_slots[i].key.data == NULL || | ||||||
|  |             string_eq(ht->key_slots[i].key, key)) { | ||||||
|             if (ht->key_slots[i].key.data != NULL) { |             if (ht->key_slots[i].key.data != NULL) { | ||||||
|                 free(ht->key_slots[i].key.data); |                 free(ht->key_slots[i].key.data); | ||||||
|             } |             } | ||||||
|             ht->key_slots[i].key = string_copy(key); |             ht->key_slots[i].key = string_copy(key); | ||||||
| 
 | 
 | ||||||
|             memcpy(ht->value_slots+i*ht->value_size, value, ht->value_size); |             memcpy(ht->value_slots + i * ht->value_size, value, ht->value_size); | ||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -75,7 +77,7 @@ bool ht_set(ed_ht *ht, string key, void *value) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void *ht_get_slot(ed_ht *ht, size_t slot) { | void *ht_get_slot(ed_ht *ht, size_t slot) { | ||||||
|     void *value = ht->value_slots+slot*ht->value_size; |     void *value = ht->value_slots + slot * ht->value_size; | ||||||
| 
 | 
 | ||||||
|     if (slot >= ht->capacity || !value) |     if (slot >= ht->capacity || !value) | ||||||
|         return NULL; |         return NULL; | ||||||
|  | @ -86,7 +88,7 @@ void *ht_get_slot(ed_ht *ht, size_t slot) { | ||||||
| void *ht_get(ed_ht *ht, string key) { | void *ht_get(ed_ht *ht, string key) { | ||||||
|     ht_hash_t hash = ht_hash(key, ht->capacity); |     ht_hash_t hash = ht_hash(key, ht->capacity); | ||||||
| 
 | 
 | ||||||
|     for (size_t i=hash; i<ht->capacity; ++i) { |     for (size_t i = hash; i < ht->capacity; ++i) { | ||||||
|         if (ht->key_slots[i].key.data == NULL) { |         if (ht->key_slots[i].key.data == NULL) { | ||||||
|             return NULL; |             return NULL; | ||||||
|         } |         } | ||||||
|  | @ -102,13 +104,15 @@ void *ht_get(ed_ht *ht, string key) { | ||||||
| void ht_remove(ed_ht *ht, string key) { | void ht_remove(ed_ht *ht, string key) { | ||||||
|     ht_hash_t hash = ht_hash(key, ht->capacity); |     ht_hash_t hash = ht_hash(key, ht->capacity); | ||||||
| 
 | 
 | ||||||
|     for (size_t i=hash; i<ht->capacity; ++i) { |     for (size_t i = hash; i < ht->capacity; ++i) { | ||||||
|         if (ht->key_slots[i].key.data == NULL) { |         if (ht->key_slots[i].key.data == NULL) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (string_eq(ht->key_slots[i].key, key)) { |         if (string_eq(ht->key_slots[i].key, key)) { | ||||||
|             ht->key_slots[i] = (ed_ht_slot) { 0 }; |             free(ht->key_slots[i].key.data); | ||||||
|  |             ht->key_slots[i] = (ed_ht_slot){0}; | ||||||
|  |             return; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										224
									
								
								src/main.c
								
								
								
								
							
							
						
						
									
										224
									
								
								src/main.c
								
								
								
								
							|  | @ -15,13 +15,14 @@ | ||||||
| #define ED_BUFFER_IMPLEMENTATION | #define ED_BUFFER_IMPLEMENTATION | ||||||
| #define ED_FILE_IO_IMPLEMENTATION | #define ED_FILE_IO_IMPLEMENTATION | ||||||
| #define CHAT_SLACK_IMPLEMENTATION | #define CHAT_SLACK_IMPLEMENTATION | ||||||
|  | #include "ui.h" | ||||||
|  | 
 | ||||||
| #include "ed_array.h" | #include "ed_array.h" | ||||||
| #include "file_io.h" | #include "file_io.h" | ||||||
| #include "gfx.h" | #include "gfx.h" | ||||||
| #include "ht.h" | #include "ht.h" | ||||||
| #include "slack_api.h" | #include "slack_api.h" | ||||||
| #include "string.h" | #include "string.h" | ||||||
| #include "ui.h" |  | ||||||
| 
 | 
 | ||||||
| // static Arena default_arena = {0};
 | // static Arena default_arena = {0};
 | ||||||
| // static Arena temporary_arena = {0};
 | // static Arena temporary_arena = {0};
 | ||||||
|  | @ -40,10 +41,20 @@ static struct { | ||||||
| 
 | 
 | ||||||
|     ui_context ui_cx; |     ui_context ui_cx; | ||||||
|     gfx_context_t *gfx_cx; |     gfx_context_t *gfx_cx; | ||||||
|  | 
 | ||||||
|  |     slack_client slack_client; | ||||||
|  |     slack_user_info slack_user_info; | ||||||
|  |     array(slack_channel) slack_channels; | ||||||
|  | 
 | ||||||
|  |     ed_ht slack_messages; | ||||||
|  |     ed_ht slack_users; | ||||||
|  | 
 | ||||||
|  |     // strictly a weak reference to a channel id
 | ||||||
|  |     string selected_channel_idx; | ||||||
| } state; | } state; | ||||||
| 
 | 
 | ||||||
| void ed_init(_gfx_frame_func frame_func) { | void init(_gfx_frame_func frame_func) { | ||||||
|     state.gfx_cx = gfx_init_context(frame_func, 640, 480); |     state.gfx_cx = gfx_init_context(frame_func, 1280, 720); | ||||||
|     state.ui_cx = ui_init_context(); |     state.ui_cx = ui_init_context(); | ||||||
| 
 | 
 | ||||||
|     // TODO: grab default font from the system
 |     // TODO: grab default font from the system
 | ||||||
|  | @ -145,51 +156,144 @@ void ed_frame(int mouse_x, int mouse_y, bool mouse_left_down, | ||||||
|                                       .mouse_right_down = mouse_right_down, |                                       .mouse_right_down = mouse_right_down, | ||||||
|                                   }); |                                   }); | ||||||
| 
 | 
 | ||||||
|     ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL, |     // TODO: make having custom title bar configurable
 | ||||||
|                ui_make_size(ui_children_sum, ui_fill), UI_FLAG_DRAW_BACKGROUND); |     { | ||||||
|  |         size_t title_bar_elm = | ||||||
|  |             ui_element(&state.ui_cx, _String("chat - [slack sux]"), | ||||||
|  |                        UI_AXIS_HORIZONTAL, ui_make_size(ui_fill, ui_exact(32)), | ||||||
|  |                        UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT | | ||||||
|  |                            UI_FLAG_CENTERED_TEXT); | ||||||
|  | 
 | ||||||
|  |         ui_interaction title_bar_interaction = | ||||||
|  |             _ui_test_interaction(&state.ui_cx, title_bar_elm); | ||||||
|  | 
 | ||||||
|  |         if (title_bar_interaction.dragging) { | ||||||
|  |             // TODO: drag the window around
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ui_push_parent(&state.ui_cx); | ||||||
|  |         { | ||||||
|  |             ui_element(&state.ui_cx, _String("title bar spacer"), | ||||||
|  |                        UI_AXIS_HORIZONTAL, ui_make_size(ui_fill, ui_exact(32)), | ||||||
|  |                        0); | ||||||
|  | 
 | ||||||
|  |             if (ui_button_centered(&state.ui_cx, _String("X")).clicked) { | ||||||
|  |                 keep_running = false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         ui_pop_parent(&state.ui_cx); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ui_element(&state.ui_cx, _String("main content"), UI_AXIS_HORIZONTAL, | ||||||
|  |                ui_make_size(ui_fill, ui_fill), 0); | ||||||
|  | 
 | ||||||
|     ui_push_parent(&state.ui_cx); |     ui_push_parent(&state.ui_cx); | ||||||
|     { |     { | ||||||
|         ui_element(&state.ui_cx, _String(buffer), UI_AXIS_HORIZONTAL, |         ui_element(&state.ui_cx, _String("channel sidebar"), UI_AXIS_VERTICAL, | ||||||
|                    ui_make_size(ui_fit_text, ui_fit_text), |                    ui_make_size(ui_children_sum, ui_fill), | ||||||
|                    UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT); |                    UI_FLAG_DRAW_BACKGROUND); | ||||||
|  |         ui_push_parent(&state.ui_cx); | ||||||
|  |         { | ||||||
|  |             ui_element(&state.ui_cx, state.slack_user_info.team, | ||||||
|  |                        UI_AXIS_HORIZONTAL, | ||||||
|  |                        ui_make_size(ui_fit_text, ui_fit_text), | ||||||
|  |                        UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT); | ||||||
| 
 | 
 | ||||||
|         if (ui_button(&state.ui_cx, _String("#dev-general")).clicked) { |             for (int i = 0; i < state.slack_channels.size; ++i) { | ||||||
|             printf("you clicked the dev-general button\n"); |                 if (ui_button(&state.ui_cx, state.slack_channels.data[i].name) | ||||||
|             state.show_thing = !state.show_thing; |                         .clicked) { | ||||||
|         } |                     state.selected_channel_idx = | ||||||
|         if (ui_button(&state.ui_cx, _String("#dev-help")).clicked) { |                         state.slack_channels.data[i].id; | ||||||
|             printf("you clicked the dev-help button\n"); |  | ||||||
|         } |  | ||||||
| 
 | 
 | ||||||
|         if (state.show_thing) { |                     // FIXME: DO NOT DO THIS ON THE UI THREAD!
 | ||||||
|             ui_interaction interaction = _ui_test_interaction( |                     if (!ht_get(&state.slack_messages, | ||||||
|                 &state.ui_cx, |                                 state.selected_channel_idx)) { | ||||||
|                 ui_element(&state.ui_cx, _String("thread list"), |  | ||||||
|                            UI_AXIS_VERTICAL, |  | ||||||
|                            ui_make_size(ui_children_sum, ui_children_sum), |  | ||||||
|                            UI_FLAG_DRAW_BACKGROUND)); |  | ||||||
| 
 | 
 | ||||||
|             ui_push_parent(&state.ui_cx); |                         array(slack_message) messages = slack_api_message_list( | ||||||
|             ui_element(&state.ui_cx, _String("List of Threads"), |                             &state.slack_client, state.selected_channel_idx); | ||||||
|                        UI_AXIS_VERTICAL, ui_make_size(ui_fit_text, ui_fit_text), | 
 | ||||||
|                        UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT | |                         ht_set(&state.slack_messages, | ||||||
|                            UI_FLAG_HOVERABLE); |                                state.selected_channel_idx, &messages); | ||||||
|             if (interaction.hovering) { |                     } | ||||||
|                 if (ui_button(&state.ui_cx, _String("Thread 1")).clicked) { |  | ||||||
|                     printf("thread 1\n"); |  | ||||||
|                 } |  | ||||||
|                 if (ui_button(&state.ui_cx, _String("Thread 2")).clicked) { |  | ||||||
|                     printf("thread 2\n"); |  | ||||||
|                 } |  | ||||||
|                 if (ui_button(&state.ui_cx, _String("Thread 3")).clicked) { |  | ||||||
|                     printf("thread 3\n"); |  | ||||||
|                 } |  | ||||||
|                 if (ui_button(&state.ui_cx, _String("Thread 4")).clicked) { |  | ||||||
|                     printf("thread 4\n"); |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             ui_pop_parent(&state.ui_cx); | 
 | ||||||
|  |             // if (ui_button(&state.ui_cx, _String("#dev-general")).clicked) {
 | ||||||
|  |             //     printf("you clicked the dev-general button\n");
 | ||||||
|  |             //     state.show_thing = !state.show_thing;
 | ||||||
|  |             // }
 | ||||||
|  |             // if (ui_button(&state.ui_cx, _String("#dev-help")).clicked) {
 | ||||||
|  |             //     printf("you clicked the dev-help button\n");
 | ||||||
|  |             // }
 | ||||||
|  | 
 | ||||||
|  |             if (state.show_thing) { | ||||||
|  |                 ui_interaction interaction = _ui_test_interaction( | ||||||
|  |                     &state.ui_cx, | ||||||
|  |                     ui_element(&state.ui_cx, _String("thread list"), | ||||||
|  |                                UI_AXIS_VERTICAL, | ||||||
|  |                                ui_make_size(ui_children_sum, ui_children_sum), | ||||||
|  |                                UI_FLAG_DRAW_BACKGROUND)); | ||||||
|  | 
 | ||||||
|  |                 ui_push_parent(&state.ui_cx); | ||||||
|  |                 ui_element(&state.ui_cx, _String("List of Threads"), | ||||||
|  |                            UI_AXIS_VERTICAL, | ||||||
|  |                            ui_make_size(ui_fit_text, ui_fit_text), | ||||||
|  |                            UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT | | ||||||
|  |                                UI_FLAG_HOVERABLE); | ||||||
|  |                 if (interaction.hovering) { | ||||||
|  |                     if (ui_button(&state.ui_cx, _String("Thread 1")).clicked) { | ||||||
|  |                         printf("thread 1\n"); | ||||||
|  |                     } | ||||||
|  |                     if (ui_button(&state.ui_cx, _String("Thread 2")).clicked) { | ||||||
|  |                         printf("thread 2\n"); | ||||||
|  |                     } | ||||||
|  |                     if (ui_button(&state.ui_cx, _String("Thread 3")).clicked) { | ||||||
|  |                         printf("thread 3\n"); | ||||||
|  |                     } | ||||||
|  |                     if (ui_button(&state.ui_cx, _String("Thread 4")).clicked) { | ||||||
|  |                         printf("thread 4\n"); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 ui_pop_parent(&state.ui_cx); | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |         ui_pop_parent(&state.ui_cx); | ||||||
|  | 
 | ||||||
|  |         ui_element(&state.ui_cx, _String("channel contents"), UI_AXIS_VERTICAL, | ||||||
|  |                    ui_make_size(ui_fill, ui_fill), UI_FLAG_DRAW_BACKGROUND); | ||||||
|  |         ui_push_parent(&state.ui_cx); | ||||||
|  |         { | ||||||
|  |             if (state.selected_channel_idx.data != NULL) { | ||||||
|  |                 array(slack_message) *messages = | ||||||
|  |                     ht_get(&state.slack_messages, state.selected_channel_idx); | ||||||
|  | 
 | ||||||
|  |                 if (messages) { | ||||||
|  |                     uint8_t text_buf[1024] = {0}; | ||||||
|  |                     for (int i = messages->size - 1; i >= 0; --i) { | ||||||
|  |                         string message_text = messages->data[i].normal.text; | ||||||
|  |                         slack_user *user = ht_get( | ||||||
|  |                             &state.slack_users, messages->data[i].normal.user); | ||||||
|  | 
 | ||||||
|  |                         if (user) { | ||||||
|  |                             snprintf(text_buf, 1024, "%.*s: %.*s", | ||||||
|  |                                      (int)user->real_name.len, user->name.data, | ||||||
|  |                                      (int)message_text.len, message_text.data); | ||||||
|  |                         } else { | ||||||
|  |                             snprintf(text_buf, 1024, "Unknown: %.*s", | ||||||
|  |                                      (int)message_text.len, message_text.data); | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                         ui_button(&state.ui_cx, _CString_To_String(text_buf)); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 ui_element(&state.ui_cx, _String("no channel selected"), | ||||||
|  |                            UI_AXIS_VERTICAL, ui_make_size(ui_fill, ui_fit_text), | ||||||
|  |                            UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT | | ||||||
|  |                                UI_FLAG_CENTERED_TEXT); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         ui_pop_parent(&state.ui_cx); | ||||||
|     } |     } | ||||||
|     ui_pop_parent(&state.ui_cx); |     ui_pop_parent(&state.ui_cx); | ||||||
| 
 | 
 | ||||||
|  | @ -203,13 +307,43 @@ void ed_frame(int mouse_x, int mouse_y, bool mouse_left_down, | ||||||
| int main(int argc, char *argv[]) { | int main(int argc, char *argv[]) { | ||||||
|     curl_global_init(CURL_GLOBAL_ALL); |     curl_global_init(CURL_GLOBAL_ALL); | ||||||
| 
 | 
 | ||||||
|     slack_client client = slack_init_client(); |     // just here to make development quicker for now
 | ||||||
|     _slack_debug_print_auth_test(&client); |     uint8_t token_buffer[4096]; | ||||||
|  |     uint8_t cookie_buffer[4096]; | ||||||
| 
 | 
 | ||||||
|     curl_global_cleanup(); |     size_t token_buffer_size = get_file_size(_String("./bin/slack_token.txt")); | ||||||
|     return 0; |     size_t cookie_buffer_size = | ||||||
|  |         get_file_size(_String("./bin/slack_cookie.txt")); | ||||||
|  |     assert("failed to open token file" && | ||||||
|  |            load_file(_String("./bin/slack_token.txt"), token_buffer_size, | ||||||
|  |                      token_buffer)); | ||||||
|  |     assert("failed to open cookie file" && | ||||||
|  |            load_file(_String("./bin/slack_cookie.txt"), cookie_buffer_size, | ||||||
|  |                      cookie_buffer)); | ||||||
| 
 | 
 | ||||||
|     ed_init(ed_frame); |     token_buffer[token_buffer_size - 1] = 0; | ||||||
|  |     cookie_buffer[cookie_buffer_size - 1] = 0; | ||||||
|  | 
 | ||||||
|  |     state.slack_client = slack_init_client(_CString_To_String(token_buffer), | ||||||
|  |                                            _CString_To_String(cookie_buffer)); | ||||||
|  |     state.slack_user_info = slack_api_auth_test(&state.slack_client); | ||||||
|  |     state.slack_channels = | ||||||
|  |         slack_api_channel_list(&state.slack_client, state.slack_user_info); | ||||||
|  |     state.selected_channel_idx = (string){0}; | ||||||
|  | 
 | ||||||
|  |     // state.slack_messages = slack_api_message_list(
 | ||||||
|  |     //     &state.slack_client, state.slack_channels.data[0].id);
 | ||||||
|  | 
 | ||||||
|  |     state.slack_messages = ht_create(1000, sizeof(array(slack_message))); | ||||||
|  |     state.slack_users = ht_create(1000, sizeof(slack_user)); | ||||||
|  |     // array(slack_user) users =
 | ||||||
|  |     //     slack_api_user_list(&state.slack_client,
 | ||||||
|  |     //     state.slack_user_info.team_id);
 | ||||||
|  |     // for (int i = 0; i < users.size; ++i) {
 | ||||||
|  |     //     ht_set(&state.slack_users, users.data[i].id, &users.data[i]);
 | ||||||
|  |     // }
 | ||||||
|  | 
 | ||||||
|  |     init(ed_frame); | ||||||
| 
 | 
 | ||||||
|     while (keep_running) { |     while (keep_running) { | ||||||
|         gfx_run_events(state.gfx_cx); |         gfx_run_events(state.gfx_cx); | ||||||
|  |  | ||||||
							
								
								
									
										472
									
								
								src/slack_api.h
								
								
								
								
							
							
						
						
									
										472
									
								
								src/slack_api.h
								
								
								
								
							|  | @ -9,13 +9,70 @@ | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     CURL *curl; |     CURL *curl; | ||||||
|  | 
 | ||||||
|  |     string token; | ||||||
|  |     string cookie; | ||||||
| } slack_client; | } slack_client; | ||||||
| 
 | 
 | ||||||
| void _slack_debug_print_auth_test(slack_client *client); | typedef struct { | ||||||
|  |     string url; | ||||||
|  |     string team; | ||||||
|  |     string user; | ||||||
|  |     string team_id; | ||||||
|  |     string user_id; | ||||||
|  | } slack_user_info; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     string id; | ||||||
|  |     string name; | ||||||
|  | 
 | ||||||
|  |     string topic; | ||||||
|  |     string purpose; | ||||||
|  | } slack_channel; | ||||||
|  | arrayTemplate(slack_channel); | ||||||
|  | 
 | ||||||
|  | typedef enum { | ||||||
|  |     SLACK_MESSAGE_TYPE_MESSAGE, | ||||||
|  | } slack_message_type_t; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     slack_message_type_t type; | ||||||
|  | } slack_message_type; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     string user; | ||||||
|  |     string text; | ||||||
|  | } slack_message_normal; | ||||||
|  | 
 | ||||||
|  | typedef struct { | ||||||
|  |     slack_message_type type; | ||||||
|  |     union { | ||||||
|  |         slack_message_normal normal; | ||||||
|  |     }; | ||||||
|  | } slack_message; | ||||||
|  | arrayTemplate(slack_message); | ||||||
|  | 
 | ||||||
|  | // TODO: merge with `slack_user_info`
 | ||||||
|  | typedef struct { | ||||||
|  |     string id; | ||||||
|  |     string name; | ||||||
|  |     string real_name; | ||||||
|  | } slack_user; | ||||||
|  | arrayTemplate(slack_user); | ||||||
|  | 
 | ||||||
|  | slack_user_info slack_api_auth_test(slack_client *client); | ||||||
|  | array(slack_channel) | ||||||
|  |     slack_api_channel_list(slack_client *client, slack_user_info user); | ||||||
|  | array(slack_message) | ||||||
|  |     slack_api_message_list(slack_client *client, string channel_id); | ||||||
|  | array(slack_user) slack_api_user_list(slack_client *client, string team_id); | ||||||
| 
 | 
 | ||||||
| slack_client slack_init_client(); | slack_client slack_init_client(); | ||||||
| 
 | 
 | ||||||
| #ifdef CHAT_SLACK_IMPLEMENTATION | #ifdef CHAT_SLACK_IMPLEMENTATION | ||||||
|  | #include <cJSON/cJSON.h> | ||||||
|  | 
 | ||||||
|  | #include <cJSON/cJSON.c> | ||||||
| 
 | 
 | ||||||
| static size_t write_memory_callback(void *contents, size_t size, size_t nmemb, | static size_t write_memory_callback(void *contents, size_t size, size_t nmemb, | ||||||
|                                     void *userp) { |                                     void *userp) { | ||||||
|  | @ -24,53 +81,444 @@ static size_t write_memory_callback(void *contents, size_t size, size_t nmemb, | ||||||
|     return real_size; |     return real_size; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void _slack_debug_print_auth_test(slack_client *client, string token, | slack_user_info slack_api_auth_test(slack_client *client) { | ||||||
|                                   string cookie) { |  | ||||||
|     if (client->curl) { |     if (client->curl) { | ||||||
|         struct curl_slist *headers = NULL; |         struct curl_slist *headers = NULL; | ||||||
|         headers = curl_slist_append(headers, "Content-Type: application/json"); |         headers = curl_slist_append( | ||||||
|         // headers = curl_slist_append(headers, cookie);
 |             headers, "Content-Type: application/x-www-form-urlencoded"); | ||||||
|  |         headers = curl_slist_append(headers, client->cookie.data); | ||||||
| 
 | 
 | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_URL, |         curl_easy_setopt(client->curl, CURLOPT_URL, | ||||||
|                          "https://slack.com/api/auth.test"); |                          "https://slack.com/api/auth.test"); | ||||||
| 
 | 
 | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER); |         curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER); | ||||||
|         // curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER, token);
 |         curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER, | ||||||
|  |                          client->token.data); | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_USERAGENT, |         curl_easy_setopt(client->curl, CURLOPT_USERAGENT, | ||||||
|                          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; " |                          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; " | ||||||
|                          "rv:76.0) Gecko/20100101 Firefox/76.0"); |                          "rv:76.0) Gecko/20100101 Firefox/76.0"); | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers); |         curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers); | ||||||
| 
 | 
 | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_POST, 1L); |         curl_easy_setopt(client->curl, CURLOPT_POST, 1L); | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, "{}"); |         curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, ""); | ||||||
| 
 | 
 | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L); |         curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L); | ||||||
| 
 | 
 | ||||||
|         array(uint8_t) chunk = newArray(uint8_t, 10000); |         array(uint8_t) chunk = newArray(uint8_t, 4096); | ||||||
| 
 | 
 | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION, |         curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION, | ||||||
|                          write_memory_callback); |                          write_memory_callback); | ||||||
|         curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk); |         curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk); | ||||||
| 
 | 
 | ||||||
|         /* Perform the request, res gets the return code */ |  | ||||||
|         CURLcode res = curl_easy_perform(client->curl); |         CURLcode res = curl_easy_perform(client->curl); | ||||||
|         /* Check for errors */ | 
 | ||||||
|         if (res != CURLE_OK) { |         if (res != CURLE_OK) { | ||||||
|             fprintf(stderr, "curl_easy_perform() failed: %s\n", |             fprintf(stderr, "curl_easy_perform() failed: %s\n", | ||||||
|                     curl_easy_strerror(res)); |                     curl_easy_strerror(res)); | ||||||
| 
 |  | ||||||
|         } else { |         } else { | ||||||
|             fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data); |             fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         // parse json
 | ||||||
|  |         cJSON *json = cJSON_ParseWithLength(chunk.data, chunk.size); | ||||||
|  |         if (json == NULL) { | ||||||
|  |             const char *error_ptr = cJSON_GetErrorPtr(); | ||||||
|  |             if (error_ptr != NULL) { | ||||||
|  |                 fprintf(stderr, | ||||||
|  |                         "SLACK CLIENT: Failed to parse /auth.test: %s\n", | ||||||
|  |                         error_ptr); | ||||||
|  | 
 | ||||||
|  |                 // FIXME: don't panic here
 | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |             const cJSON *ok = cJSON_GetObjectItemCaseSensitive(json, "ok"); | ||||||
|  |             if (cJSON_IsBool(ok) && ok->valueint == 1) { | ||||||
|  |                 fprintf(stderr, "SLACK CLIENT: /auth.test succeeded!\n"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *url = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "url"); | ||||||
|  |                 const cJSON *team = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "team"); | ||||||
|  |                 const cJSON *user = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "user"); | ||||||
|  |                 const cJSON *team_id = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "team_id"); | ||||||
|  |                 const cJSON *user_id = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "user_id"); | ||||||
|  | 
 | ||||||
|  |                 return (slack_user_info){ | ||||||
|  |                     .url = string_copy_cstring(url->valuestring), | ||||||
|  |                     .team = string_copy_cstring(team->valuestring), | ||||||
|  |                     .user = string_copy_cstring(user->valuestring), | ||||||
|  |                     .team_id = string_copy_cstring(team_id->valuestring), | ||||||
|  |                     .user_id = string_copy_cstring(user_id->valuestring), | ||||||
|  |                 }; | ||||||
|  |             } else { | ||||||
|  |                 const cJSON *error = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "error"); | ||||||
|  |                 if (cJSON_IsString(error)) { | ||||||
|  |                     fprintf(stderr, "SLACK CLIENT: /auth.test failed: %s\n", | ||||||
|  |                             error->valuestring); | ||||||
|  |                 } else { | ||||||
|  |                     fprintf(stderr, "SLACK CLIENT: /auth.test failed: failed " | ||||||
|  |                                     "to parse 'error' field\n"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         cJSON_Delete(json); | ||||||
|  | 
 | ||||||
|         free(chunk.data); |         free(chunk.data); | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | array(slack_channel) | ||||||
|  |     slack_api_channel_list(slack_client *client, slack_user_info user) { | ||||||
|  |     if (client->curl) { | ||||||
|  |         struct curl_slist *headers = NULL; | ||||||
|  |         headers = curl_slist_append( | ||||||
|  |             headers, "Content-Type: application/x-www-form-urlencoded"); | ||||||
|  |         headers = curl_slist_append(headers, client->cookie.data); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_URL, | ||||||
|  |                          "https://slack.com/api/conversations.list"); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER, | ||||||
|  |                          client->token.data); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_USERAGENT, | ||||||
|  |                          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; " | ||||||
|  |                          "rv:76.0) Gecko/20100101 Firefox/76.0"); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POST, 1L); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, "limit=10"); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L); | ||||||
|  | 
 | ||||||
|  |         array(uint8_t) chunk = newArray(uint8_t, 4096); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION, | ||||||
|  |                          write_memory_callback); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk); | ||||||
|  | 
 | ||||||
|  |         CURLcode res = curl_easy_perform(client->curl); | ||||||
|  | 
 | ||||||
|  |         if (res != CURLE_OK) { | ||||||
|  |             fprintf(stderr, "curl_easy_perform() failed: %s\n", | ||||||
|  |                     curl_easy_strerror(res)); | ||||||
|  |         } else { | ||||||
|  |             fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // parse json
 | ||||||
|  |         cJSON *json = cJSON_ParseWithLength(chunk.data, chunk.size); | ||||||
|  |         if (json == NULL) { | ||||||
|  |             const char *error_ptr = cJSON_GetErrorPtr(); | ||||||
|  |             if (error_ptr != NULL) { | ||||||
|  |                 fprintf( | ||||||
|  |                     stderr, | ||||||
|  |                     "SLACK CLIENT: Failed to parse /conversations.list: %s\n", | ||||||
|  |                     error_ptr); | ||||||
|  | 
 | ||||||
|  |                 // FIXME: don't panic here
 | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |             const cJSON *ok = cJSON_GetObjectItemCaseSensitive(json, "ok"); | ||||||
|  |             if (cJSON_IsBool(ok) && ok->valueint == 1) { | ||||||
|  |                 array(slack_channel) channels = newArray(slack_channel, 10); | ||||||
|  | 
 | ||||||
|  |                 fprintf(stderr, | ||||||
|  |                         "SLACK CLIENT: /conversations.list succeeded!\n"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_channels = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "channels"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_channel; | ||||||
|  |                 cJSON_ArrayForEach(json_channel, json_channels) { | ||||||
|  |                     const cJSON *id = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_channel, "id"); | ||||||
|  |                     const cJSON *name = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_channel, "name"); | ||||||
|  |                     const cJSON *topic = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_channel, "topic"); | ||||||
|  |                     const cJSON *purpose = cJSON_GetObjectItemCaseSensitive( | ||||||
|  |                         json_channel, "purpose"); | ||||||
|  | 
 | ||||||
|  |                     slack_channel channel = { | ||||||
|  |                         .id = string_copy_cstring(id->valuestring), | ||||||
|  |                         .name = string_copy_cstring(name->valuestring), | ||||||
|  |                         .topic = string_copy_cstring(topic->valuestring), | ||||||
|  |                         .purpose = string_copy_cstring(purpose->valuestring), | ||||||
|  |                     }; | ||||||
|  | 
 | ||||||
|  |                     pushArray(slack_channel, &channels, channel); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 return channels; | ||||||
|  |             } else { | ||||||
|  |                 const cJSON *error = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "error"); | ||||||
|  |                 if (cJSON_IsString(error)) { | ||||||
|  |                     fprintf(stderr, | ||||||
|  |                             "SLACK CLIENT: /conversations.list failed: %s\n", | ||||||
|  |                             error->valuestring); | ||||||
|  |                 } else { | ||||||
|  |                     fprintf(stderr, | ||||||
|  |                             "SLACK CLIENT: /conversations.list failed: failed " | ||||||
|  |                             "to parse 'error' field\n"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         cJSON_Delete(json); | ||||||
|  | 
 | ||||||
|  |         free(chunk.data); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // TODO: create some slack api error type
 | ||||||
|  |     return (array(slack_channel)){0}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | array(slack_message) | ||||||
|  |     slack_api_message_list(slack_client *client, string channel_id) { | ||||||
|  | 
 | ||||||
|  |     if (client->curl) { | ||||||
|  |         struct curl_slist *headers = NULL; | ||||||
|  |         headers = curl_slist_append( | ||||||
|  |             headers, "Content-Type: application/x-www-form-urlencoded"); | ||||||
|  |         headers = curl_slist_append(headers, client->cookie.data); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_URL, | ||||||
|  |                          "https://slack.com/api/conversations.history"); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER, | ||||||
|  |                          client->token.data); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_USERAGENT, | ||||||
|  |                          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; " | ||||||
|  |                          "rv:76.0) Gecko/20100101 Firefox/76.0"); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POST, 1L); | ||||||
|  | 
 | ||||||
|  |         char *fields_buffer[2048]; | ||||||
|  |         snprintf(fields_buffer, 2048, "channel=%.*s&limit=10", | ||||||
|  |                  (int)channel_id.len, channel_id.data); | ||||||
|  |         fprintf(stderr, "fields_buffer: %s\n", fields_buffer); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, fields_buffer); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L); | ||||||
|  | 
 | ||||||
|  |         array(uint8_t) chunk = newArray(uint8_t, 4096); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION, | ||||||
|  |                          write_memory_callback); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk); | ||||||
|  | 
 | ||||||
|  |         CURLcode res = curl_easy_perform(client->curl); | ||||||
|  | 
 | ||||||
|  |         if (res != CURLE_OK) { | ||||||
|  |             fprintf(stderr, "curl_easy_perform() failed: %s\n", | ||||||
|  |                     curl_easy_strerror(res)); | ||||||
|  |         } else { | ||||||
|  |             fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // parse json
 | ||||||
|  |         cJSON *json = cJSON_ParseWithLength(chunk.data, chunk.size); | ||||||
|  |         if (json == NULL) { | ||||||
|  |             const char *error_ptr = cJSON_GetErrorPtr(); | ||||||
|  |             if (error_ptr != NULL) { | ||||||
|  |                 fprintf(stderr, | ||||||
|  |                         "SLACK CLIENT: Failed to parse /conversations.history: " | ||||||
|  |                         "%s\n", | ||||||
|  |                         error_ptr); | ||||||
|  | 
 | ||||||
|  |                 // FIXME: don't panic here
 | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |             const cJSON *ok = cJSON_GetObjectItemCaseSensitive(json, "ok"); | ||||||
|  |             if (cJSON_IsBool(ok) && ok->valueint == 1) { | ||||||
|  |                 array(slack_message) messages = newArray(slack_message, 10); | ||||||
|  | 
 | ||||||
|  |                 fprintf(stderr, | ||||||
|  |                         "SLACK CLIENT: /conversations.history succeeded!\n"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_messages = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "messages"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_message; | ||||||
|  |                 cJSON_ArrayForEach(json_message, json_messages) { | ||||||
|  |                     const cJSON *type = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_message, "type"); | ||||||
|  |                     const cJSON *user = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_message, "user"); | ||||||
|  |                     const cJSON *text = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_message, "text"); | ||||||
|  | 
 | ||||||
|  |                     if (strcmp(type->valuestring, "message") == 0) { | ||||||
|  |                         slack_message message = { | ||||||
|  |                             .type = SLACK_MESSAGE_TYPE_MESSAGE, | ||||||
|  |                             .normal = | ||||||
|  |                                 (slack_message_normal){ | ||||||
|  |                                     .user = | ||||||
|  |                                         string_copy_cstring(user->valuestring), | ||||||
|  |                                     .text = | ||||||
|  |                                         string_copy_cstring(text->valuestring), | ||||||
|  |                                 }, | ||||||
|  |                         }; | ||||||
|  | 
 | ||||||
|  |                         pushArray(slack_message, &messages, message); | ||||||
|  |                     } else { | ||||||
|  |                         fprintf(stderr, | ||||||
|  |                                 "SLACK CLIENT: unknown message type: %s\n", | ||||||
|  |                                 type->valuestring); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 return messages; | ||||||
|  |             } else { | ||||||
|  |                 const cJSON *error = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "error"); | ||||||
|  |                 if (cJSON_IsString(error)) { | ||||||
|  |                     fprintf(stderr, | ||||||
|  |                             "SLACK CLIENT: /conversations.history failed: %s\n", | ||||||
|  |                             error->valuestring); | ||||||
|  |                 } else { | ||||||
|  |                     fprintf( | ||||||
|  |                         stderr, | ||||||
|  |                         "SLACK CLIENT: /conversations.history failed: failed " | ||||||
|  |                         "to parse 'error' field\n"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         cJSON_Delete(json); | ||||||
|  | 
 | ||||||
|  |         free(chunk.data); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // TODO: create some slack api error type
 | ||||||
|  |     return (array(slack_message)){0}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | array(slack_user) slack_api_user_list(slack_client *client, string team_id) { | ||||||
|  |     if (client->curl) { | ||||||
|  |         struct curl_slist *headers = NULL; | ||||||
|  |         headers = curl_slist_append( | ||||||
|  |             headers, "Content-Type: application/x-www-form-urlencoded"); | ||||||
|  |         headers = curl_slist_append(headers, client->cookie.data); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_URL, | ||||||
|  |                          "https://slack.com/api/users.list"); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPAUTH, (long)CURLAUTH_BEARER); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_XOAUTH2_BEARER, | ||||||
|  |                          client->token.data); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_USERAGENT, | ||||||
|  |                          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.15; " | ||||||
|  |                          "rv:76.0) Gecko/20100101 Firefox/76.0"); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_HTTPHEADER, headers); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POST, 1L); | ||||||
|  | 
 | ||||||
|  |         char *fields_buffer[2048]; | ||||||
|  |         snprintf(fields_buffer, 2048, "team_id=%.*s", (int)team_id.len, | ||||||
|  |                  team_id.data); | ||||||
|  |         fprintf(stderr, "fields_buffer: %s\n", fields_buffer); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_POSTFIELDS, fields_buffer); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_FOLLOWLOCATION, 1L); | ||||||
|  | 
 | ||||||
|  |         // TODO: don't allocate this on every request
 | ||||||
|  |         array(uint8_t) chunk = newArray(uint8_t, 4096); | ||||||
|  | 
 | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEFUNCTION, | ||||||
|  |                          write_memory_callback); | ||||||
|  |         curl_easy_setopt(client->curl, CURLOPT_WRITEDATA, (void *)&chunk); | ||||||
|  | 
 | ||||||
|  |         CURLcode res = curl_easy_perform(client->curl); | ||||||
|  | 
 | ||||||
|  |         if (res != CURLE_OK) { | ||||||
|  |             fprintf(stderr, "curl_easy_perform() failed: %s\n", | ||||||
|  |                     curl_easy_strerror(res)); | ||||||
|  |         } else { | ||||||
|  |             fprintf(stderr, "Got data\n%.*s\n", (int)chunk.size, chunk.data); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // parse json
 | ||||||
|  |         cJSON *json = cJSON_ParseWithLength(chunk.data, chunk.size); | ||||||
|  |         if (json == NULL) { | ||||||
|  |             const char *error_ptr = cJSON_GetErrorPtr(); | ||||||
|  |             if (error_ptr != NULL) { | ||||||
|  |                 fprintf(stderr, | ||||||
|  |                         "SLACK CLIENT: Failed to parse /users.list: %s\n", | ||||||
|  |                         error_ptr); | ||||||
|  | 
 | ||||||
|  |                 // FIXME: don't panic here
 | ||||||
|  |                 exit(1); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         { | ||||||
|  |             const cJSON *ok = cJSON_GetObjectItemCaseSensitive(json, "ok"); | ||||||
|  |             if (cJSON_IsBool(ok) && ok->valueint == 1) { | ||||||
|  |                 array(slack_user) users = newArray(slack_user, 10); | ||||||
|  | 
 | ||||||
|  |                 fprintf(stderr, "SLACK CLIENT: /users.list succeeded!\n"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_users = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "members"); | ||||||
|  | 
 | ||||||
|  |                 const cJSON *json_user; | ||||||
|  |                 cJSON_ArrayForEach(json_user, json_users) { | ||||||
|  |                     const cJSON *user_id = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_user, "id"); | ||||||
|  |                     const cJSON *name = | ||||||
|  |                         cJSON_GetObjectItemCaseSensitive(json_user, "name"); | ||||||
|  |                     const cJSON *real_name = cJSON_GetObjectItemCaseSensitive( | ||||||
|  |                         json_user, "real_name"); | ||||||
|  | 
 | ||||||
|  |                     slack_user user = { | ||||||
|  |                         .id = string_copy_cstring(user_id->valuestring), | ||||||
|  |                         .name = string_copy_cstring(name->valuestring), | ||||||
|  |                         .real_name = | ||||||
|  |                             string_copy_cstring(real_name->valuestring), | ||||||
|  |                     }; | ||||||
|  | 
 | ||||||
|  |                     pushArray(slack_user, &users, user); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 return users; | ||||||
|  |             } else { | ||||||
|  |                 const cJSON *error = | ||||||
|  |                     cJSON_GetObjectItemCaseSensitive(json, "error"); | ||||||
|  |                 if (cJSON_IsString(error)) { | ||||||
|  |                     fprintf(stderr, "SLACK CLIENT: /users.list failed: %s\n", | ||||||
|  |                             error->valuestring); | ||||||
|  |                 } else { | ||||||
|  |                     fprintf(stderr, "SLACK CLIENT: /users.list failed: failed " | ||||||
|  |                                     "to parse 'error' field\n"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         cJSON_Delete(json); | ||||||
|  | 
 | ||||||
|  |         free(chunk.data); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // TODO: create some slack api error type
 | ||||||
|  |     return (array(slack_user)){0}; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| slack_client slack_init_client(string token, string cookie) { | slack_client slack_init_client(string token, string cookie) { | ||||||
|     CURL *curl = curl_easy_init(); |     CURL *curl = curl_easy_init(); | ||||||
| 
 | 
 | ||||||
|     slack_client client = (slack_client){.curl = curl}; |     slack_client client = | ||||||
|  |         (slack_client){.curl = curl, .token = token, .cookie = cookie}; | ||||||
| 
 | 
 | ||||||
|     return client; |     return client; | ||||||
| } | } | ||||||
|  |  | ||||||
							
								
								
									
										47
									
								
								src/string.h
								
								
								
								
							
							
						
						
									
										47
									
								
								src/string.h
								
								
								
								
							|  | @ -8,13 +8,21 @@ | ||||||
| 
 | 
 | ||||||
| #define _String(text)                                                          \ | #define _String(text)                                                          \ | ||||||
|     ((string){.data = (uint8_t *)text, .len = sizeof(text), .owned = false}) |     ((string){.data = (uint8_t *)text, .len = sizeof(text), .owned = false}) | ||||||
|  | #define _CString_To_String(text)                                               \ | ||||||
|  |     ((string){.data = (uint8_t *)text, .len = strlen(text), .owned = false}) | ||||||
| typedef struct { | typedef struct { | ||||||
|     uint8_t *data; |     uint8_t *data; | ||||||
|     size_t len; |     size_t len; | ||||||
| 
 | 
 | ||||||
|  |     // FIXME: this is so terribly bad please don't do this
 | ||||||
|     bool owned; |     bool owned; | ||||||
| } string; | } string; | ||||||
| 
 | 
 | ||||||
|  | bool string_eq(string a, string b); | ||||||
|  | bool string_eq_cstring(string a, const char *b); | ||||||
|  | string string_copy(string s); | ||||||
|  | string string_copy_cstring(const char *str); | ||||||
|  | 
 | ||||||
| #ifdef ED_STRING_IMPLEMENTATION | #ifdef ED_STRING_IMPLEMENTATION | ||||||
| bool string_eq(string a, string b) { | bool string_eq(string a, string b) { | ||||||
|     if (a.len != b.len) |     if (a.len != b.len) | ||||||
|  | @ -28,6 +36,27 @@ bool string_eq(string a, string b) { | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | bool string_eq_cstring(string a, const char *b) { | ||||||
|  |     if (b == NULL) { | ||||||
|  |         if (a.len == 0) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     size_t b_len = strlen(b); | ||||||
|  |     if (a.len != b_len) | ||||||
|  |         return false; | ||||||
|  | 
 | ||||||
|  |     for (size_t i = 0; i < a.len; ++i) { | ||||||
|  |         if (a.data[i] != b[i]) | ||||||
|  |             return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| string string_copy(string a) { | string string_copy(string a) { | ||||||
|     string new_string; |     string new_string; | ||||||
| 
 | 
 | ||||||
|  | @ -40,5 +69,23 @@ string string_copy(string a) { | ||||||
|     return new_string; |     return new_string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | string string_copy_cstring(const char *str) { | ||||||
|  |     if (str == NULL) { | ||||||
|  |         return (string){.data = NULL, .len = 0, .owned = false}; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     string new_string; | ||||||
|  | 
 | ||||||
|  |     size_t len = strlen(str); | ||||||
|  | 
 | ||||||
|  |     new_string.data = malloc(len * sizeof(uint8_t)); | ||||||
|  |     new_string.len = len; | ||||||
|  |     new_string.owned = true; | ||||||
|  | 
 | ||||||
|  |     memcpy(new_string.data, str, new_string.len * sizeof(uint8_t)); | ||||||
|  | 
 | ||||||
|  |     return new_string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #endif | #endif | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
							
								
								
									
										65
									
								
								src/ui.h
								
								
								
								
							
							
						
						
									
										65
									
								
								src/ui.h
								
								
								
								
							|  | @ -3,11 +3,12 @@ | ||||||
| #ifndef ED_UI_INCLUDED | #ifndef ED_UI_INCLUDED | ||||||
| #define ED_UI_INCLUDED | #define ED_UI_INCLUDED | ||||||
| 
 | 
 | ||||||
|  | #include "string.h" | ||||||
| #define MAX_UI_ELEMENTS 8192 | #define MAX_UI_ELEMENTS 8192 | ||||||
| 
 | 
 | ||||||
| // TODO: replace this with functions
 | // TODO: replace this with functions
 | ||||||
| #define _FONT_WIDTH 12 |  | ||||||
| #define _FONT_HEIGHT 24 | #define _FONT_HEIGHT 24 | ||||||
|  | #define _FONT_WIDTH _FONT_HEIGHT / 2 | ||||||
| 
 | 
 | ||||||
| #define _elm(index) (cx->frame_elements.data + index) | #define _elm(index) (cx->frame_elements.data + index) | ||||||
| #define _flags(index, flgs) ((_elm(index)->flags & (flgs)) == (flgs)) | #define _flags(index, flgs) ((_elm(index)->flags & (flgs)) == (flgs)) | ||||||
|  | @ -61,6 +62,9 @@ typedef struct { | ||||||
|     ((ui_semantic_size){.type = UI_SEMANTIC_SIZE_CHILDREN_SUM}) |     ((ui_semantic_size){.type = UI_SEMANTIC_SIZE_CHILDREN_SUM}) | ||||||
| #define ui_exact(value)                                                        \ | #define ui_exact(value)                                                        \ | ||||||
|     ((ui_semantic_size){.type = UI_SEMANTIC_SIZE_EXACT, .integer = value}) |     ((ui_semantic_size){.type = UI_SEMANTIC_SIZE_EXACT, .integer = value}) | ||||||
|  | #define ui_percent_of_parent(value)                                            \ | ||||||
|  |     ((ui_semantic_size){.type = UI_SEMANTIC_SIZE_PERCENT_OF_PARENT,            \ | ||||||
|  |                         .integer = value}) | ||||||
| 
 | 
 | ||||||
| typedef struct { | typedef struct { | ||||||
|     ui_axis axis; |     ui_axis axis; | ||||||
|  | @ -83,15 +87,16 @@ typedef struct { | ||||||
| } ui_element_cache_data; | } ui_element_cache_data; | ||||||
| 
 | 
 | ||||||
| typedef enum { | typedef enum { | ||||||
|     UI_FLAG_CLICKABLE = 0b000000001, |     UI_FLAG_CLICKABLE = 0b0000000001, | ||||||
|     UI_FLAG_HOVERABLE = 0b000000010, |     UI_FLAG_HOVERABLE = 0b0000000010, | ||||||
|     UI_FLAG_SCROLLABLE = 0b000000100, |     UI_FLAG_SCROLLABLE = 0b0000000100, | ||||||
|     UI_FLAG_DRAW_TEXT = 0b000001000, |     UI_FLAG_DRAW_TEXT = 0b0000001000, | ||||||
|     UI_FLAG_DRAW_BORDER = 0b000010000, |     UI_FLAG_CENTERED_TEXT = 0b1000000000, | ||||||
|     UI_FLAG_DRAW_BACKGROUND = 0b000100000, |     UI_FLAG_DRAW_BORDER = 0b0000010000, | ||||||
|     UI_FLAG_ROUNDED_BORDER = 0b001000000, |     UI_FLAG_DRAW_BACKGROUND = 0b0000100000, | ||||||
|     UI_FLAG_FLOATING = 0b010000000, |     UI_FLAG_ROUNDED_BORDER = 0b0001000000, | ||||||
|     UI_FLAG_CUSTOM_DRAW_FUNC = 0b100000000, |     UI_FLAG_FLOATING = 0b0010000000, | ||||||
|  |     UI_FLAG_CUSTOM_DRAW_FUNC = 0b0100000000, | ||||||
| } ui_flags; | } ui_flags; | ||||||
| 
 | 
 | ||||||
| // Ephemeral frame only UI Element data
 | // Ephemeral frame only UI Element data
 | ||||||
|  | @ -161,7 +166,8 @@ ui_context ui_init_context() { | ||||||
|         .prev = -1, |         .prev = -1, | ||||||
|         .parent = -1, |         .parent = -1, | ||||||
|         .size = { |         .size = { | ||||||
|             .axis = UI_AXIS_HORIZONTAL, |             // TODO: make this configurable
 | ||||||
|  |             .axis = UI_AXIS_VERTICAL, | ||||||
|             .computed_size = {640, 480}, |             .computed_size = {640, 480}, | ||||||
|         }}; |         }}; | ||||||
|     pushArray(ui_element_frame_data, &frame_elements, frame_data); |     pushArray(ui_element_frame_data, &frame_elements, frame_data); | ||||||
|  | @ -190,10 +196,10 @@ size_t ui_element(ui_context *cx, string label, ui_axis axis, | ||||||
|                   ui_semantic_size size[2], ui_flags flags) { |                   ui_semantic_size size[2], ui_flags flags) { | ||||||
|     ui_element_frame_data frame_data = (ui_element_frame_data){ |     ui_element_frame_data frame_data = (ui_element_frame_data){ | ||||||
|         .index = cx->frame_elements.size, |         .index = cx->frame_elements.size, | ||||||
|         // TODO: don't just set this to label, because then elements
 |         // TODO: don't just set `key` to label, because then elements
 | ||||||
|         // with the same label can't be created together
 |         // with the same label can't be created together
 | ||||||
|         .key = label, |         .key = string_copy(label), | ||||||
|         .label = label, |         .label = string_copy(label), | ||||||
|         .first = -1, |         .first = -1, | ||||||
|         .last = -1, |         .last = -1, | ||||||
|         .next = -1, |         .next = -1, | ||||||
|  | @ -218,7 +224,7 @@ size_t ui_element(ui_context *cx, string label, ui_axis axis, | ||||||
|     } else { |     } else { | ||||||
|         bool did_insert = ht_set(&cx->cached_elements, label, |         bool did_insert = ht_set(&cx->cached_elements, label, | ||||||
|                                  &(ui_element_cache_data){ |                                  &(ui_element_cache_data){ | ||||||
|                                      .label = label, |                                      .label = string_copy(label), | ||||||
|                                      .size = {0}, |                                      .size = {0}, | ||||||
|                                      .last_instantiated_index = cx->frame_index, |                                      .last_instantiated_index = cx->frame_index, | ||||||
|                                  }); |                                  }); | ||||||
|  | @ -267,6 +273,15 @@ ui_interaction ui_button(ui_context *cx, string label) { | ||||||
|     return _ui_test_interaction(cx, id); |     return _ui_test_interaction(cx, id); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | ui_interaction ui_button_centered(ui_context *cx, string label) { | ||||||
|  |     size_t id = ui_element( | ||||||
|  |         cx, label, UI_AXIS_HORIZONTAL, ui_make_size(ui_fit_text, ui_fit_text), | ||||||
|  |         UI_FLAG_DRAW_BACKGROUND | UI_FLAG_DRAW_TEXT | UI_FLAG_CENTERED_TEXT | | ||||||
|  |             UI_FLAG_HOVERABLE | UI_FLAG_CLICKABLE); | ||||||
|  | 
 | ||||||
|  |     return _ui_test_interaction(cx, id); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static uint32_t _ui_ancestor_size(ui_context *cx, size_t element_index, | static uint32_t _ui_ancestor_size(ui_context *cx, size_t element_index, | ||||||
|                                   ui_axis axis) { |                                   ui_axis axis) { | ||||||
|     if (element_index == SIZE_MAX || _parent(element_index) == SIZE_MAX) { |     if (element_index == SIZE_MAX || _parent(element_index) == SIZE_MAX) { | ||||||
|  | @ -293,9 +308,9 @@ static void _ui_compute_simple_layout(ui_context *cx, | ||||||
|     switch (elm->size.semantic_size[axis].type) { |     switch (elm->size.semantic_size[axis].type) { | ||||||
|     case UI_SEMANTIC_SIZE_FIT_TEXT: |     case UI_SEMANTIC_SIZE_FIT_TEXT: | ||||||
|         if (axis == UI_AXIS_HORIZONTAL) { |         if (axis == UI_AXIS_HORIZONTAL) { | ||||||
|             elm->size.computed_size[axis] = elm->label.len * _FONT_WIDTH; |             elm->size.computed_size[axis] = (elm->label.len + 2) * _FONT_WIDTH; | ||||||
|         } else if (axis == UI_AXIS_VERTICAL) { |         } else if (axis == UI_AXIS_VERTICAL) { | ||||||
|             elm->size.computed_size[axis] = _FONT_HEIGHT; |             elm->size.computed_size[axis] = _FONT_HEIGHT + (_FONT_HEIGHT / 4); | ||||||
|         } |         } | ||||||
|         break; |         break; | ||||||
| 
 | 
 | ||||||
|  | @ -504,7 +519,16 @@ void ui_render(ui_context *cx, _ui_render_text_func text_func, | ||||||
|         ui_element_frame_data *elm = &cx->frame_elements.data[i]; |         ui_element_frame_data *elm = &cx->frame_elements.data[i]; | ||||||
| 
 | 
 | ||||||
|         if (_flags(i, UI_FLAG_DRAW_TEXT)) { |         if (_flags(i, UI_FLAG_DRAW_TEXT)) { | ||||||
|             text_func(text, (float[]){(float)elm->size.computed_pos[0], |             uint32_t x_offset = 0; | ||||||
|  | 
 | ||||||
|  |             if (_flags(i, UI_FLAG_CENTERED_TEXT)) { | ||||||
|  |                 uint32_t half_width = elm->size.computed_size[0] / 2; | ||||||
|  |                 uint32_t half_text_width = (text.len / 2) * _FONT_WIDTH; | ||||||
|  |                 x_offset = half_width - half_text_width; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             text_func(text, (float[]){(float)elm->size.computed_pos[0] + | ||||||
|  |                                           (float)x_offset, | ||||||
|                                       (float)elm->size.computed_pos[1]}); |                                       (float)elm->size.computed_pos[1]}); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  | @ -537,6 +561,7 @@ void ui_prune(ui_context *cx) { | ||||||
|                 // %zu, frame index: %zu\n", (int)key.len, key.data,
 |                 // %zu, frame index: %zu\n", (int)key.len, key.data,
 | ||||||
|                 // cached->last_instantiated_index, cx->frame_index);
 |                 // cached->last_instantiated_index, cx->frame_index);
 | ||||||
| 
 | 
 | ||||||
|  |                 free(cached->label.data); | ||||||
|                 ht_remove(&cx->cached_elements, key); |                 ht_remove(&cx->cached_elements, key); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -546,7 +571,9 @@ void ui_prune(ui_context *cx) { | ||||||
|     do { |     do { | ||||||
|         __auto_type elm = _elm(child_index); |         __auto_type elm = _elm(child_index); | ||||||
|         if (elm->label.owned) { |         if (elm->label.owned) { | ||||||
|             free(elm->label.data); |             // FIXME: deal with potential memory leaks (really just use an temp
 | ||||||
|  |             // allocator)
 | ||||||
|  |             // free(elm->label.data);
 | ||||||
|         } |         } | ||||||
|     } while ((child_index = _next(child_index)) < SIZE_MAX); |     } while ((child_index = _next(child_index)) < SIZE_MAX); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,173 @@ | ||||||
|  | /* Generated by wayland-scanner 1.22.0 */ | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Copyright © 2008-2013 Kristian Høgsberg | ||||||
|  |  * Copyright © 2013      Rafael Antognolli | ||||||
|  |  * Copyright © 2013      Jasper St. Pierre | ||||||
|  |  * Copyright © 2010-2013 Intel Corporation | ||||||
|  |  * Copyright © 2015-2017 Samsung Electronics Co., Ltd | ||||||
|  |  * Copyright © 2015-2017 Red Hat Inc. | ||||||
|  |  * | ||||||
|  |  * Permission is hereby granted, free of charge, to any person obtaining a | ||||||
|  |  * copy of this software and associated documentation files (the "Software"), | ||||||
|  |  * to deal in the Software without restriction, including without limitation | ||||||
|  |  * the rights to use, copy, modify, merge, publish, distribute, sublicense, | ||||||
|  |  * and/or sell copies of the Software, and to permit persons to whom the | ||||||
|  |  * Software is furnished to do so, subject to the following conditions: | ||||||
|  |  * | ||||||
|  |  * The above copyright notice and this permission notice (including the next | ||||||
|  |  * paragraph) shall be included in all copies or substantial portions of the | ||||||
|  |  * Software. | ||||||
|  |  * | ||||||
|  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL | ||||||
|  |  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||||||
|  |  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | ||||||
|  |  * DEALINGS IN THE SOFTWARE. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <stdint.h> | ||||||
|  | #include "wayland-util.h" | ||||||
|  | 
 | ||||||
|  | extern const struct wl_interface wl_output_interface; | ||||||
|  | extern const struct wl_interface wl_seat_interface; | ||||||
|  | extern const struct wl_interface wl_surface_interface; | ||||||
|  | extern const struct wl_interface xdg_popup_interface; | ||||||
|  | extern const struct wl_interface xdg_positioner_interface; | ||||||
|  | extern const struct wl_interface xdg_surface_interface; | ||||||
|  | extern const struct wl_interface xdg_toplevel_interface; | ||||||
|  | 
 | ||||||
|  | static const struct wl_interface *xdg_shell_types[] = { | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	&xdg_positioner_interface, | ||||||
|  | 	&xdg_surface_interface, | ||||||
|  | 	&wl_surface_interface, | ||||||
|  | 	&xdg_toplevel_interface, | ||||||
|  | 	&xdg_popup_interface, | ||||||
|  | 	&xdg_surface_interface, | ||||||
|  | 	&xdg_positioner_interface, | ||||||
|  | 	&xdg_toplevel_interface, | ||||||
|  | 	&wl_seat_interface, | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	&wl_seat_interface, | ||||||
|  | 	NULL, | ||||||
|  | 	&wl_seat_interface, | ||||||
|  | 	NULL, | ||||||
|  | 	NULL, | ||||||
|  | 	&wl_output_interface, | ||||||
|  | 	&wl_seat_interface, | ||||||
|  | 	NULL, | ||||||
|  | 	&xdg_positioner_interface, | ||||||
|  | 	NULL, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_wm_base_requests[] = { | ||||||
|  | 	{ "destroy", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "create_positioner", "n", xdg_shell_types + 4 }, | ||||||
|  | 	{ "get_xdg_surface", "no", xdg_shell_types + 5 }, | ||||||
|  | 	{ "pong", "u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_wm_base_events[] = { | ||||||
|  | 	{ "ping", "u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | WL_EXPORT const struct wl_interface xdg_wm_base_interface = { | ||||||
|  | 	"xdg_wm_base", 6, | ||||||
|  | 	4, xdg_wm_base_requests, | ||||||
|  | 	1, xdg_wm_base_events, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_positioner_requests[] = { | ||||||
|  | 	{ "destroy", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_size", "ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_anchor_rect", "iiii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_anchor", "u", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_gravity", "u", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_constraint_adjustment", "u", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_offset", "ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_reactive", "3", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_parent_size", "3ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_parent_configure", "3u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | WL_EXPORT const struct wl_interface xdg_positioner_interface = { | ||||||
|  | 	"xdg_positioner", 6, | ||||||
|  | 	10, xdg_positioner_requests, | ||||||
|  | 	0, NULL, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_surface_requests[] = { | ||||||
|  | 	{ "destroy", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "get_toplevel", "n", xdg_shell_types + 7 }, | ||||||
|  | 	{ "get_popup", "n?oo", xdg_shell_types + 8 }, | ||||||
|  | 	{ "set_window_geometry", "iiii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "ack_configure", "u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_surface_events[] = { | ||||||
|  | 	{ "configure", "u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | WL_EXPORT const struct wl_interface xdg_surface_interface = { | ||||||
|  | 	"xdg_surface", 6, | ||||||
|  | 	5, xdg_surface_requests, | ||||||
|  | 	1, xdg_surface_events, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_toplevel_requests[] = { | ||||||
|  | 	{ "destroy", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_parent", "?o", xdg_shell_types + 11 }, | ||||||
|  | 	{ "set_title", "s", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_app_id", "s", xdg_shell_types + 0 }, | ||||||
|  | 	{ "show_window_menu", "ouii", xdg_shell_types + 12 }, | ||||||
|  | 	{ "move", "ou", xdg_shell_types + 16 }, | ||||||
|  | 	{ "resize", "ouu", xdg_shell_types + 18 }, | ||||||
|  | 	{ "set_max_size", "ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_min_size", "ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_maximized", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "unset_maximized", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_fullscreen", "?o", xdg_shell_types + 21 }, | ||||||
|  | 	{ "unset_fullscreen", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "set_minimized", "", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_toplevel_events[] = { | ||||||
|  | 	{ "configure", "iia", xdg_shell_types + 0 }, | ||||||
|  | 	{ "close", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "configure_bounds", "4ii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "wm_capabilities", "5a", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | WL_EXPORT const struct wl_interface xdg_toplevel_interface = { | ||||||
|  | 	"xdg_toplevel", 6, | ||||||
|  | 	14, xdg_toplevel_requests, | ||||||
|  | 	4, xdg_toplevel_events, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_popup_requests[] = { | ||||||
|  | 	{ "destroy", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "grab", "ou", xdg_shell_types + 22 }, | ||||||
|  | 	{ "reposition", "3ou", xdg_shell_types + 24 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct wl_message xdg_popup_events[] = { | ||||||
|  | 	{ "configure", "iiii", xdg_shell_types + 0 }, | ||||||
|  | 	{ "popup_done", "", xdg_shell_types + 0 }, | ||||||
|  | 	{ "repositioned", "3u", xdg_shell_types + 0 }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | WL_EXPORT const struct wl_interface xdg_popup_interface = { | ||||||
|  | 	"xdg_popup", 6, | ||||||
|  | 	3, xdg_popup_requests, | ||||||
|  | 	3, xdg_popup_events, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							|  | @ -0,0 +1,300 @@ | ||||||
|  | /*
 | ||||||
|  |   Copyright (c) 2009-2017 Dave Gamble and cJSON contributors | ||||||
|  | 
 | ||||||
|  |   Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  |   of this software and associated documentation files (the "Software"), to deal | ||||||
|  |   in the Software without restriction, including without limitation the rights | ||||||
|  |   to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  |   copies of the Software, and to permit persons to whom the Software is | ||||||
|  |   furnished to do so, subject to the following conditions: | ||||||
|  | 
 | ||||||
|  |   The above copyright notice and this permission notice shall be included in | ||||||
|  |   all copies or substantial portions of the Software. | ||||||
|  | 
 | ||||||
|  |   THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  |   IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  |   FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  |   AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  |   LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  |   OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||||
|  |   THE SOFTWARE. | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | #ifndef cJSON__h | ||||||
|  | #define cJSON__h | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | extern "C" | ||||||
|  | { | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #if !defined(__WINDOWS__) && (defined(WIN32) || defined(WIN64) || defined(_MSC_VER) || defined(_WIN32)) | ||||||
|  | #define __WINDOWS__ | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifdef __WINDOWS__ | ||||||
|  | 
 | ||||||
|  | /* When compiling for windows, we specify a specific calling convention to avoid issues where we are being called from a project with a different default calling convention.  For windows you have 3 define options:
 | ||||||
|  | 
 | ||||||
|  | CJSON_HIDE_SYMBOLS - Define this in the case where you don't want to ever dllexport symbols | ||||||
|  | CJSON_EXPORT_SYMBOLS - Define this on library build when you want to dllexport symbols (default) | ||||||
|  | CJSON_IMPORT_SYMBOLS - Define this if you want to dllimport symbol | ||||||
|  | 
 | ||||||
|  | For *nix builds that support visibility attribute, you can define similar behavior by | ||||||
|  | 
 | ||||||
|  | setting default visibility to hidden by adding | ||||||
|  | -fvisibility=hidden (for gcc) | ||||||
|  | or | ||||||
|  | -xldscope=hidden (for sun cc) | ||||||
|  | to CFLAGS | ||||||
|  | 
 | ||||||
|  | then using the CJSON_API_VISIBILITY flag to "export" the same symbols the way CJSON_EXPORT_SYMBOLS does | ||||||
|  | 
 | ||||||
|  | */ | ||||||
|  | 
 | ||||||
|  | #define CJSON_CDECL __cdecl | ||||||
|  | #define CJSON_STDCALL __stdcall | ||||||
|  | 
 | ||||||
|  | /* export symbols by default, this is necessary for copy pasting the C and header file */ | ||||||
|  | #if !defined(CJSON_HIDE_SYMBOLS) && !defined(CJSON_IMPORT_SYMBOLS) && !defined(CJSON_EXPORT_SYMBOLS) | ||||||
|  | #define CJSON_EXPORT_SYMBOLS | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #if defined(CJSON_HIDE_SYMBOLS) | ||||||
|  | #define CJSON_PUBLIC(type)   type CJSON_STDCALL | ||||||
|  | #elif defined(CJSON_EXPORT_SYMBOLS) | ||||||
|  | #define CJSON_PUBLIC(type)   __declspec(dllexport) type CJSON_STDCALL | ||||||
|  | #elif defined(CJSON_IMPORT_SYMBOLS) | ||||||
|  | #define CJSON_PUBLIC(type)   __declspec(dllimport) type CJSON_STDCALL | ||||||
|  | #endif | ||||||
|  | #else /* !__WINDOWS__ */ | ||||||
|  | #define CJSON_CDECL | ||||||
|  | #define CJSON_STDCALL | ||||||
|  | 
 | ||||||
|  | #if (defined(__GNUC__) || defined(__SUNPRO_CC) || defined (__SUNPRO_C)) && defined(CJSON_API_VISIBILITY) | ||||||
|  | #define CJSON_PUBLIC(type)   __attribute__((visibility("default"))) type | ||||||
|  | #else | ||||||
|  | #define CJSON_PUBLIC(type) type | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /* project version */ | ||||||
|  | #define CJSON_VERSION_MAJOR 1 | ||||||
|  | #define CJSON_VERSION_MINOR 7 | ||||||
|  | #define CJSON_VERSION_PATCH 17 | ||||||
|  | 
 | ||||||
|  | #include <stddef.h> | ||||||
|  | 
 | ||||||
|  | /* cJSON Types: */ | ||||||
|  | #define cJSON_Invalid (0) | ||||||
|  | #define cJSON_False  (1 << 0) | ||||||
|  | #define cJSON_True   (1 << 1) | ||||||
|  | #define cJSON_NULL   (1 << 2) | ||||||
|  | #define cJSON_Number (1 << 3) | ||||||
|  | #define cJSON_String (1 << 4) | ||||||
|  | #define cJSON_Array  (1 << 5) | ||||||
|  | #define cJSON_Object (1 << 6) | ||||||
|  | #define cJSON_Raw    (1 << 7) /* raw json */ | ||||||
|  | 
 | ||||||
|  | #define cJSON_IsReference 256 | ||||||
|  | #define cJSON_StringIsConst 512 | ||||||
|  | 
 | ||||||
|  | /* The cJSON structure: */ | ||||||
|  | typedef struct cJSON | ||||||
|  | { | ||||||
|  |     /* next/prev allow you to walk array/object chains. Alternatively, use GetArraySize/GetArrayItem/GetObjectItem */ | ||||||
|  |     struct cJSON *next; | ||||||
|  |     struct cJSON *prev; | ||||||
|  |     /* An array or object item will have a child pointer pointing to a chain of the items in the array/object. */ | ||||||
|  |     struct cJSON *child; | ||||||
|  | 
 | ||||||
|  |     /* The type of the item, as above. */ | ||||||
|  |     int type; | ||||||
|  | 
 | ||||||
|  |     /* The item's string, if type==cJSON_String  and type == cJSON_Raw */ | ||||||
|  |     char *valuestring; | ||||||
|  |     /* writing to valueint is DEPRECATED, use cJSON_SetNumberValue instead */ | ||||||
|  |     int valueint; | ||||||
|  |     /* The item's number, if type==cJSON_Number */ | ||||||
|  |     double valuedouble; | ||||||
|  | 
 | ||||||
|  |     /* The item's name string, if this item is the child of, or is in the list of subitems of an object. */ | ||||||
|  |     char *string; | ||||||
|  | } cJSON; | ||||||
|  | 
 | ||||||
|  | typedef struct cJSON_Hooks | ||||||
|  | { | ||||||
|  |       /* malloc/free are CDECL on Windows regardless of the default calling convention of the compiler, so ensure the hooks allow passing those functions directly. */ | ||||||
|  |       void *(CJSON_CDECL *malloc_fn)(size_t sz); | ||||||
|  |       void (CJSON_CDECL *free_fn)(void *ptr); | ||||||
|  | } cJSON_Hooks; | ||||||
|  | 
 | ||||||
|  | typedef int cJSON_bool; | ||||||
|  | 
 | ||||||
|  | /* Limits how deeply nested arrays/objects can be before cJSON rejects to parse them.
 | ||||||
|  |  * This is to prevent stack overflows. */ | ||||||
|  | #ifndef CJSON_NESTING_LIMIT | ||||||
|  | #define CJSON_NESTING_LIMIT 1000 | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | /* returns the version of cJSON as a string */ | ||||||
|  | CJSON_PUBLIC(const char*) cJSON_Version(void); | ||||||
|  | 
 | ||||||
|  | /* Supply malloc, realloc and free functions to cJSON */ | ||||||
|  | CJSON_PUBLIC(void) cJSON_InitHooks(cJSON_Hooks* hooks); | ||||||
|  | 
 | ||||||
|  | /* Memory Management: the caller is always responsible to free the results from all variants of cJSON_Parse (with cJSON_Delete) and cJSON_Print (with stdlib free, cJSON_Hooks.free_fn, or cJSON_free as appropriate). The exception is cJSON_PrintPreallocated, where the caller has full responsibility of the buffer. */ | ||||||
|  | /* Supply a block of JSON, and this returns a cJSON object you can interrogate. */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_Parse(const char *value); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_ParseWithLength(const char *value, size_t buffer_length); | ||||||
|  | /* ParseWithOpts allows you to require (and check) that the JSON is null terminated, and to retrieve the pointer to the final byte parsed. */ | ||||||
|  | /* If you supply a ptr in return_parse_end and parsing fails, then return_parse_end will contain a pointer to the error so will match cJSON_GetErrorPtr(). */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_ParseWithOpts(const char *value, const char **return_parse_end, cJSON_bool require_null_terminated); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_ParseWithLengthOpts(const char *value, size_t buffer_length, const char **return_parse_end, cJSON_bool require_null_terminated); | ||||||
|  | 
 | ||||||
|  | /* Render a cJSON entity to text for transfer/storage. */ | ||||||
|  | CJSON_PUBLIC(char *) cJSON_Print(const cJSON *item); | ||||||
|  | /* Render a cJSON entity to text for transfer/storage without any formatting. */ | ||||||
|  | CJSON_PUBLIC(char *) cJSON_PrintUnformatted(const cJSON *item); | ||||||
|  | /* Render a cJSON entity to text using a buffered strategy. prebuffer is a guess at the final size. guessing well reduces reallocation. fmt=0 gives unformatted, =1 gives formatted */ | ||||||
|  | CJSON_PUBLIC(char *) cJSON_PrintBuffered(const cJSON *item, int prebuffer, cJSON_bool fmt); | ||||||
|  | /* Render a cJSON entity to text using a buffer already allocated in memory with given length. Returns 1 on success and 0 on failure. */ | ||||||
|  | /* NOTE: cJSON is not always 100% accurate in estimating how much memory it will use, so to be safe allocate 5 bytes more than you actually need */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_PrintPreallocated(cJSON *item, char *buffer, const int length, const cJSON_bool format); | ||||||
|  | /* Delete a cJSON entity and all subentities. */ | ||||||
|  | CJSON_PUBLIC(void) cJSON_Delete(cJSON *item); | ||||||
|  | 
 | ||||||
|  | /* Returns the number of items in an array (or object). */ | ||||||
|  | CJSON_PUBLIC(int) cJSON_GetArraySize(const cJSON *array); | ||||||
|  | /* Retrieve item number "index" from array "array". Returns NULL if unsuccessful. */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_GetArrayItem(const cJSON *array, int index); | ||||||
|  | /* Get item "string" from object. Case insensitive. */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_GetObjectItem(const cJSON * const object, const char * const string); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_GetObjectItemCaseSensitive(const cJSON * const object, const char * const string); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_HasObjectItem(const cJSON *object, const char *string); | ||||||
|  | /* For analysing failed parses. This returns a pointer to the parse error. You'll probably need to look a few chars back to make sense of it. Defined when cJSON_Parse() returns 0. 0 when cJSON_Parse() succeeds. */ | ||||||
|  | CJSON_PUBLIC(const char *) cJSON_GetErrorPtr(void); | ||||||
|  | 
 | ||||||
|  | /* Check item type and return its value */ | ||||||
|  | CJSON_PUBLIC(char *) cJSON_GetStringValue(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(double) cJSON_GetNumberValue(const cJSON * const item); | ||||||
|  | 
 | ||||||
|  | /* These functions check the type of an item */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsInvalid(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsFalse(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsTrue(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsBool(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsNull(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsNumber(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsString(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsArray(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsObject(const cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_IsRaw(const cJSON * const item); | ||||||
|  | 
 | ||||||
|  | /* These calls create a cJSON item of the appropriate type. */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateNull(void); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateTrue(void); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateFalse(void); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateBool(cJSON_bool boolean); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateNumber(double num); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateString(const char *string); | ||||||
|  | /* raw json */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateRaw(const char *raw); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateArray(void); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateObject(void); | ||||||
|  | 
 | ||||||
|  | /* Create a string where valuestring references a string so
 | ||||||
|  |  * it will not be freed by cJSON_Delete */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateStringReference(const char *string); | ||||||
|  | /* Create an object/array that only references it's elements so
 | ||||||
|  |  * they will not be freed by cJSON_Delete */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateObjectReference(const cJSON *child); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateArrayReference(const cJSON *child); | ||||||
|  | 
 | ||||||
|  | /* These utilities create an Array of count items.
 | ||||||
|  |  * The parameter count cannot be greater than the number of elements in the number array, otherwise array access will be out of bounds.*/ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateIntArray(const int *numbers, int count); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateFloatArray(const float *numbers, int count); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateDoubleArray(const double *numbers, int count); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_CreateStringArray(const char *const *strings, int count); | ||||||
|  | 
 | ||||||
|  | /* Append item to the specified array/object. */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToArray(cJSON *array, cJSON *item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item); | ||||||
|  | /* Use this when string is definitely const (i.e. a literal, or as good as), and will definitely survive the cJSON object.
 | ||||||
|  |  * WARNING: When this function was used, make sure to always check that (item->type & cJSON_StringIsConst) is zero before | ||||||
|  |  * writing to `item->string` */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObjectCS(cJSON *object, const char *string, cJSON *item); | ||||||
|  | /* Append reference to item to the specified array/object. Use this when you want to add an existing cJSON to a new cJSON, but don't want to corrupt your existing cJSON. */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_AddItemReferenceToArray(cJSON *array, cJSON *item); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_AddItemReferenceToObject(cJSON *object, const char *string, cJSON *item); | ||||||
|  | 
 | ||||||
|  | /* Remove/Detach items from Arrays/Objects. */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_DetachItemViaPointer(cJSON *parent, cJSON * const item); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromArray(cJSON *array, int which); | ||||||
|  | CJSON_PUBLIC(void) cJSON_DeleteItemFromArray(cJSON *array, int which); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromObject(cJSON *object, const char *string); | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromObjectCaseSensitive(cJSON *object, const char *string); | ||||||
|  | CJSON_PUBLIC(void) cJSON_DeleteItemFromObject(cJSON *object, const char *string); | ||||||
|  | CJSON_PUBLIC(void) cJSON_DeleteItemFromObjectCaseSensitive(cJSON *object, const char *string); | ||||||
|  | 
 | ||||||
|  | /* Update array items. */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_InsertItemInArray(cJSON *array, int which, cJSON *newitem); /* Shifts pre-existing items to the right. */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemViaPointer(cJSON * const parent, cJSON * const item, cJSON * replacement); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemInArray(cJSON *array, int which, cJSON *newitem); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemInObject(cJSON *object,const char *string,cJSON *newitem); | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemInObjectCaseSensitive(cJSON *object,const char *string,cJSON *newitem); | ||||||
|  | 
 | ||||||
|  | /* Duplicate a cJSON item */ | ||||||
|  | CJSON_PUBLIC(cJSON *) cJSON_Duplicate(const cJSON *item, cJSON_bool recurse); | ||||||
|  | /* Duplicate will create a new, identical cJSON item to the one you pass, in new memory that will
 | ||||||
|  |  * need to be released. With recurse!=0, it will duplicate any children connected to the item. | ||||||
|  |  * The item->next and ->prev pointers are always zero on return from Duplicate. */ | ||||||
|  | /* Recursively compare two cJSON items for equality. If either a or b is NULL or invalid, they will be considered unequal.
 | ||||||
|  |  * case_sensitive determines if object keys are treated case sensitive (1) or case insensitive (0) */ | ||||||
|  | CJSON_PUBLIC(cJSON_bool) cJSON_Compare(const cJSON * const a, const cJSON * const b, const cJSON_bool case_sensitive); | ||||||
|  | 
 | ||||||
|  | /* Minify a strings, remove blank characters(such as ' ', '\t', '\r', '\n') from strings.
 | ||||||
|  |  * The input pointer json cannot point to a read-only address area, such as a string constant,  | ||||||
|  |  * but should point to a readable and writable address area. */ | ||||||
|  | CJSON_PUBLIC(void) cJSON_Minify(char *json); | ||||||
|  | 
 | ||||||
|  | /* Helper functions for creating and adding items to an object at the same time.
 | ||||||
|  |  * They return the added item or NULL on failure. */ | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddNullToObject(cJSON * const object, const char * const name); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddTrueToObject(cJSON * const object, const char * const name); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddFalseToObject(cJSON * const object, const char * const name); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddBoolToObject(cJSON * const object, const char * const name, const cJSON_bool boolean); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddNumberToObject(cJSON * const object, const char * const name, const double number); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddStringToObject(cJSON * const object, const char * const name, const char * const string); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddRawToObject(cJSON * const object, const char * const name, const char * const raw); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddObjectToObject(cJSON * const object, const char * const name); | ||||||
|  | CJSON_PUBLIC(cJSON*) cJSON_AddArrayToObject(cJSON * const object, const char * const name); | ||||||
|  | 
 | ||||||
|  | /* When assigning an integer value, it needs to be propagated to valuedouble too. */ | ||||||
|  | #define cJSON_SetIntValue(object, number) ((object) ? (object)->valueint = (object)->valuedouble = (number) : (number)) | ||||||
|  | /* helper for the cJSON_SetNumberValue macro */ | ||||||
|  | CJSON_PUBLIC(double) cJSON_SetNumberHelper(cJSON *object, double number); | ||||||
|  | #define cJSON_SetNumberValue(object, number) ((object != NULL) ? cJSON_SetNumberHelper(object, (double)number) : (number)) | ||||||
|  | /* Change the valuestring of a cJSON_String object, only takes effect when type of object is cJSON_String */ | ||||||
|  | CJSON_PUBLIC(char*) cJSON_SetValuestring(cJSON *object, const char *valuestring); | ||||||
|  | 
 | ||||||
|  | /* If the object is not a boolean type this does nothing and returns cJSON_Invalid else it returns the new type*/ | ||||||
|  | #define cJSON_SetBoolValue(object, boolValue) ( \ | ||||||
|  |     (object != NULL && ((object)->type & (cJSON_False|cJSON_True))) ? \ | ||||||
|  |     (object)->type=((object)->type &(~(cJSON_False|cJSON_True)))|((boolValue)?cJSON_True:cJSON_False) : \ | ||||||
|  |     cJSON_Invalid\ | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | /* Macro for iterating over an array or object */ | ||||||
|  | #define cJSON_ArrayForEach(element, array) for(element = (array != NULL) ? (array)->child : NULL; element != NULL; element = element->next) | ||||||
|  | 
 | ||||||
|  | /* malloc/free objects using the malloc/free functions that have been set with cJSON_InitHooks */ | ||||||
|  | CJSON_PUBLIC(void *) cJSON_malloc(size_t size); | ||||||
|  | CJSON_PUBLIC(void) cJSON_free(void *object); | ||||||
|  | 
 | ||||||
|  | #ifdef __cplusplus | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
		Loading…
	
		Reference in New Issue