UNIXworkcode

1 /* 2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. 3 * 4 * Copyright 2017 Olaf Wintermann. All rights reserved. 5 * 6 * Redistribution and use in source and binary forms, with or without 7 * modification, are permitted provided that the following conditions are met: 8 * 9 * 1. Redistributions of source code must retain the above copyright 10 * notice, this list of conditions and the following disclaimer. 11 * 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 26 * POSSIBILITY OF SUCH DAMAGE. 27 */ 28 29 #include "utils.h" 30 #include "properties.h" 31 32 #include <stdio.h> 33 #include <string.h> 34 35 #include <cx/string.h> 36 #include <cx/buffer.h> 37 38 UiPathElm* ui_default_pathelm_func(const char* full_path, size_t len, size_t* ret_nelm, void* data) { 39 cxstring *pathelms; 40 size_t nelm = cx_strsplit_a(cxDefaultAllocator, cx_strn(full_path, len), CX_STR("/"), 4096, &pathelms); 41 42 if (nelm == 0) { 43 *ret_nelm = 0; 44 return NULL; 45 } 46 47 UiPathElm* elms = (UiPathElm*)calloc(nelm, sizeof(UiPathElm)); 48 size_t n = nelm; 49 int j = 0; 50 for (int i = 0; i < nelm; i++) { 51 cxstring c = pathelms[i]; 52 if (c.length == 0) { 53 if (i == 0) { 54 c.length = 1; 55 } 56 else { 57 n--; 58 continue; 59 } 60 } 61 62 cxmutstr m = cx_strdup(c); 63 elms[j].name = m.ptr; 64 elms[j].name_len = m.length; 65 66 size_t elm_path_len = c.ptr + c.length - full_path; 67 cxmutstr elm_path = cx_strdup(cx_strn(full_path, elm_path_len)); 68 elms[j].path = elm_path.ptr; 69 elms[j].path_len = elm_path.length; 70 71 j++; 72 } 73 *ret_nelm = n; 74 75 return elms; 76 } 77 78 void ui_get_window_default_width(int *width, int *height) { 79 const char *width_str = ui_get_property("ui.window.width"); 80 const char *height_str = ui_get_property("ui.window.height"); 81 if(width_str && height_str) { 82 int w = atoi(width_str); 83 int h = atoi(height_str); 84 if(w > 0 && h > 0) { 85 *width = w; 86 *height = h; 87 } 88 } 89 } 90 91 // from UCX json.c 92 static cxmutstr escape_string(cxstring str, bool escape_slash) { 93 // note: this function produces the string without enclosing quotes 94 // the reason is that we don't want to allocate memory just for that 95 CxBuffer buf = {0}; 96 97 bool all_printable = true; 98 for (size_t i = 0; i < str.length; i++) { 99 unsigned char c = str.ptr[i]; 100 bool escape = c < 0x20 || c == '\\' || c == '""' 101 || (escape_slash && c == '/'); 102 103 if (all_printable && escape) { 104 size_t capa = str.length + 32; 105 char *space = cxMallocDefault(capa); 106 if (space == NULL) return cx_mutstrn(NULL, 0); 107 cxBufferInit(&buf, space, capa, NULL, CX_BUFFER_AUTO_EXTEND); 108 cxBufferWrite(str.ptr, 1, i, &buf); 109 all_printable = false; 110 } 111 if (escape) { 112 cxBufferPut(&buf, '\\'); 113 if (c == '\"') { 114 cxBufferPut(&buf, '\"'); 115 } else if (c == '\n') { 116 cxBufferPut(&buf, 'n'); 117 } else if (c == '\t') { 118 cxBufferPut(&buf, 't'); 119 } else if (c == '\r') { 120 cxBufferPut(&buf, 'r'); 121 } else if (c == '\\') { 122 cxBufferPut(&buf, '\\'); 123 } else if (c == '/') { 124 cxBufferPut(&buf, '/'); 125 } else if (c == '\f') { 126 cxBufferPut(&buf, 'f'); 127 } else if (c == '\b') { 128 cxBufferPut(&buf, 'b'); 129 } else { 130 char code[6]; 131 snprintf(code, sizeof(code), "u%04x", (unsigned int) c); 132 cxBufferPutString(&buf, code); 133 } 134 } else if (!all_printable) { 135 cxBufferPut(&buf, c); 136 } 137 } 138 cxmutstr ret; 139 if (all_printable) { 140 // don't copy the string when we don't need to escape anything 141 ret = cx_mutstrn((char*)str.ptr, str.length); 142 } else { 143 ret = cx_mutstrn(buf.space, buf.size); 144 } 145 cxBufferDestroy(&buf); 146 return ret; 147 } 148 149 cxmutstr ui_escape_string(cxstring str) { 150 return escape_string(str, FALSE); 151 } 152