930 lines
30 KiB
C++
930 lines
30 KiB
C++
//
|
|
// Fluid C++ Code Writer code for the Fast Light Tool Kit (FLTK).
|
|
//
|
|
// Copyright 1998-2025 by Bill Spitzak and others.
|
|
//
|
|
// This library is free software. Distribution and use rights are outlined in
|
|
// the file "COPYING" which should have been included with this file. If this
|
|
// file is missing or damaged, see the license at:
|
|
//
|
|
// https://www.fltk.org/COPYING.php
|
|
//
|
|
// Please see the following page on how to report bugs and issues:
|
|
//
|
|
// https://www.fltk.org/bugs.php
|
|
//
|
|
|
|
#include "io/Code_Writer.h"
|
|
|
|
#include "Fluid.h"
|
|
#include "Project.h"
|
|
#include "proj/mergeback.h"
|
|
#include "nodes/Window_Node.h"
|
|
#include "nodes/Function_Node.h"
|
|
|
|
#include <FL/filename.H>
|
|
#include "../src/flstring.h"
|
|
|
|
#include <zlib.h>
|
|
|
|
#include <string>
|
|
#include <iomanip>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
|
|
using namespace fld;
|
|
using namespace fld::io;
|
|
using namespace fld::proj;
|
|
|
|
/**
|
|
Return true if c can be in a C identifier.
|
|
I needed this so it is not messed up by locale settings.
|
|
\param[in] c a character, or the start of a utf-8 sequence
|
|
\return 1 if c is alphanumeric or '_'
|
|
*/
|
|
int is_id(char c) {
|
|
return (c>='a' && c<='z') || (c>='A' && c<='Z') || (c>='0' && c<='9') || c=='_';
|
|
}
|
|
|
|
/** \brief Return a unique name for the given object.
|
|
|
|
This function combines the name and label into an identifier. It then checks
|
|
if that id was already taken by another object, and if so, appends a
|
|
hexadecimal value which is incremented until the id is unique in this file.
|
|
|
|
If a new id was created, it is stored in the id tree.
|
|
|
|
\param[in] o create an ID for this object
|
|
\param[in] type is the first word of the ID
|
|
\param[in] name if name is set, it is appended to the ID
|
|
\param[in] label else if label is set, it is appended, skipping non-keyword characters
|
|
\return buffer to a unique identifier, managed by Code_Writer, so caller must NOT free() it
|
|
*/
|
|
const char* Code_Writer::unique_id(void* o, const char* type, const char* name, const char* label) {
|
|
char buffer[128];
|
|
char* q = buffer;
|
|
char* q_end = q + 128 - 8 - 1; // room for hex number and NUL
|
|
while (*type) *q++ = *type++;
|
|
*q++ = '_';
|
|
const char* n = name;
|
|
if (!n || !*n) n = label;
|
|
if (n && *n) {
|
|
while (*n && !is_id(*n)) n++;
|
|
while (is_id(*n) && (q < q_end)) *q++ = *n++;
|
|
}
|
|
*q = 0;
|
|
// okay, search the tree and see if the name was already used:
|
|
int which = 0;
|
|
for (;;) {
|
|
auto it = unique_id_list.find(buffer);
|
|
// If the id does not exist, add it to the map
|
|
if (it == unique_id_list.end()) {
|
|
it = unique_id_list.insert(std::make_pair(buffer, o)).first;
|
|
return it->first.c_str();
|
|
}
|
|
// If it does exist, and the pointers are the same, just return it.
|
|
if (it->second == o) {
|
|
return it->first.c_str();
|
|
}
|
|
// Else repeat until we have a new id,
|
|
sprintf(q,"%x",++which);
|
|
}
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////
|
|
// return current indentation:
|
|
|
|
|
|
/**
|
|
Return a C string that indents code to the given depth.
|
|
|
|
Indentation can be changed by modifying the multiplicator (``*2`` to keep
|
|
the FLTK indent style). Changing `spaces` to a list of tabs would generate
|
|
tab indents instead. This function can also be used for fixed depth indents
|
|
in the header file.
|
|
|
|
Do *not* ever make this a user preference, or you will end up writing a
|
|
fully featured code formatter.
|
|
|
|
\param[in] set generate this indent depth
|
|
\return pointer to a static string
|
|
*/
|
|
const char *Code_Writer::indent(int set) {
|
|
static const char* spaces = " ";
|
|
int i = set * 2;
|
|
if (i>64) i = 64;
|
|
if (i<0) i = 0;
|
|
return spaces+64-i;
|
|
}
|
|
|
|
/**
|
|
Return a C string that indents code to the current source file depth.
|
|
\return pointer to a static string
|
|
*/
|
|
const char *Code_Writer::indent() {
|
|
return indent(indentation);
|
|
}
|
|
|
|
/**
|
|
Return a C string that indents code to the current source file depth plus an offset.
|
|
\param[in] offset adds a temporary offset for this call only; this does not
|
|
change the `indentation` variable; offset can be negative
|
|
\return pointer to a static string
|
|
*/
|
|
const char *Code_Writer::indent_plus(int offset) {
|
|
return indent(indentation+offset);
|
|
}
|
|
|
|
|
|
/**
|
|
Print a formatted line to the header file, unless the same line was produced before in this header file.
|
|
\note Resulting line is cropped at 1023 bytes.
|
|
\param[in] format printf-style formatting text, followed by a vararg list
|
|
\return 1 if the text was written to the file, 0 if it was previously written.
|
|
*/
|
|
int Code_Writer::write_h_once(const char *format, ...) {
|
|
va_list args;
|
|
char buf[1024];
|
|
va_start(args, format);
|
|
vsnprintf(buf, sizeof(buf), format, args);
|
|
va_end(args);
|
|
if (text_in_header.find(buf) != text_in_header.end()) {
|
|
return 0;
|
|
}
|
|
fprintf(header_file, "%s\n", buf);
|
|
text_in_header.insert(buf);
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
Print a formatted line to the source file, unless the same line was produced before in this code file.
|
|
\note Resulting line is cropped at 1023 bytes.
|
|
\param[in] format printf-style formatting text, followed by a vararg list
|
|
\return 1 if the text was written to the file, 0 if it was previously written.
|
|
*/
|
|
int Code_Writer::write_c_once(const char *format, ...) {
|
|
va_list args;
|
|
char buf[1024];
|
|
va_start(args, format);
|
|
vsnprintf(buf, sizeof(buf), format, args);
|
|
va_end(args);
|
|
// Return if the text was already printed to the header file.
|
|
if (text_in_header.find(buf) != text_in_header.end()) {
|
|
return 0;
|
|
}
|
|
// Return if the text was already printed to the source file.
|
|
if (text_in_code.find(buf) != text_in_code.end()) {
|
|
return 0;
|
|
}
|
|
crc_printf("%s\n", buf);
|
|
text_in_code.insert(buf);
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
Return true if this pointer was already included in the code file.
|
|
If it was not, add it to the list and return false.
|
|
\param[in] pp ay pointer
|
|
\return true if found in the tree, false if added to the tree
|
|
*/
|
|
bool Code_Writer::c_contains(void *pp) {
|
|
if (ptr_in_code.find(pp) != ptr_in_code.end()) {
|
|
return true;
|
|
}
|
|
ptr_in_code.insert(pp);
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
Write a C string to the code file, escaping non-ASCII characters.
|
|
|
|
Text is broken into lines of 78 character.
|
|
FLUID " before and after every line text.
|
|
|
|
A list of control characters and ", ', and \\ are escaped by adding a \\ in
|
|
front of them. Escape ?? by writing ?\\?. All other characters that are not
|
|
between 32 and 126 inclusive will be escaped as octal characters.
|
|
|
|
This function is utf8 agnostic.
|
|
|
|
\param[in] s write this string
|
|
\param[in] length write so many bytes in this string
|
|
|
|
\see f.write_cstring(const char*)
|
|
*/
|
|
void Code_Writer::write_cstring(const char *s, int length) {
|
|
const char *next_line = "\"\n\"";
|
|
if (varused_test) {
|
|
varused = 1;
|
|
return;
|
|
}
|
|
// if we are rendering to the source code preview window, and the text is
|
|
// longer than four lines, we only render a placeholder.
|
|
if (write_codeview && ((s==nullptr) || (length>300))) {
|
|
if (length>=0)
|
|
crc_printf("\" ... %d bytes of text... \"", length);
|
|
else
|
|
crc_puts("\" ... text... \"");
|
|
return;
|
|
}
|
|
if (length==-1 || s==nullptr) {
|
|
crc_puts("\n#error string not found\n");
|
|
crc_puts("\" ... undefined size text... \"");
|
|
return;
|
|
}
|
|
|
|
const char *p = s;
|
|
const char *e = s+length;
|
|
int linelength = 1;
|
|
crc_putc('\"');
|
|
for (; p < e;) {
|
|
int c = *p++;
|
|
switch (c) {
|
|
case '\b': c = 'b'; goto QUOTED;
|
|
case '\t': c = 't'; goto QUOTED;
|
|
case '\n': c = 'n'; goto QUOTED;
|
|
case '\f': c = 'f'; goto QUOTED;
|
|
case '\r': c = 'r'; goto QUOTED;
|
|
case '\"':
|
|
case '\'':
|
|
case '\\':
|
|
QUOTED:
|
|
if (linelength >= 77) { crc_puts(next_line); linelength = 0; }
|
|
crc_putc('\\');
|
|
crc_putc(c);
|
|
linelength += 2;
|
|
break;
|
|
case '?': // prevent trigraphs by writing ?? as ?\?
|
|
if (p-2 >= s && *(p-2) == '?') goto QUOTED;
|
|
// else fall through:
|
|
default:
|
|
if (c >= ' ' && c < 127) {
|
|
// a legal ASCII character
|
|
if (linelength >= 78) { crc_puts(next_line); linelength = 0; }
|
|
crc_putc(c);
|
|
linelength++;
|
|
break;
|
|
}
|
|
// if the UTF-8 option is checked, write unicode characters verbatim
|
|
if (proj_.utf8_in_src && (c&0x80)) {
|
|
if ((c&0x40)) {
|
|
// This is the first character in a utf-8 sequence (0b11......).
|
|
// A line break would be ok here. Do not put linebreak in front of
|
|
// following characters (0b10......)
|
|
if (linelength >= 78) { crc_puts(next_line); linelength = 0; }
|
|
}
|
|
crc_putc(c);
|
|
linelength++;
|
|
break;
|
|
}
|
|
// otherwise we must print it as an octal constant:
|
|
c &= 255;
|
|
if (linelength >= 74) { crc_puts(next_line); linelength = 0; }
|
|
crc_printf("\\%03o", c);
|
|
linelength += 4;
|
|
break;
|
|
}
|
|
}
|
|
crc_putc('\"');
|
|
}
|
|
|
|
/**
|
|
Write a C string, escaping non-ASCII characters.
|
|
\param[in] s write this string
|
|
\see f.write_cstring(const char*, int)
|
|
*/
|
|
void Code_Writer::write_cstring(const char *s) {
|
|
write_cstring(s, (int)strlen(s));
|
|
}
|
|
|
|
/**
|
|
Write an array of C binary data (does not add a null).
|
|
The output is bracketed in { and }. The content is written
|
|
as decimal bytes, i.e. `{ 1, 2, 200 }`
|
|
|
|
\param[in] s a block of binary data, interpreted as unsigned bytes
|
|
\param[in] length size of the block in bytes
|
|
*/
|
|
void Code_Writer::write_cdata(const char *s, int length) {
|
|
if (varused_test) {
|
|
varused = 1;
|
|
return;
|
|
}
|
|
if (write_codeview) {
|
|
if (length>=0)
|
|
crc_printf("{ /* ... %d bytes of binary data... */ }", length);
|
|
else
|
|
crc_puts("{ /* ... binary data... */ }");
|
|
return;
|
|
}
|
|
if (length==-1) {
|
|
crc_puts("\n#error data not found\n");
|
|
crc_puts("{ /* ... undefined size binary data... */ }");
|
|
return;
|
|
}
|
|
const unsigned char *w = (const unsigned char *)s;
|
|
const unsigned char *e = w+length;
|
|
int linelength = 1;
|
|
crc_putc('{');
|
|
for (; w < e;) {
|
|
unsigned char c = *w++;
|
|
if (c>99) linelength += 4;
|
|
else if (c>9) linelength += 3;
|
|
else linelength += 2;
|
|
if (linelength >= 77) {crc_puts("\n"); linelength = 0;}
|
|
crc_printf("%d", c);
|
|
if (w<e) crc_putc(',');
|
|
}
|
|
crc_putc('}');
|
|
}
|
|
|
|
/**
|
|
Print a formatted line to the source file.
|
|
\param[in] format printf-style formatting text
|
|
\param[in] args list of arguments
|
|
*/
|
|
void Code_Writer::vwrite_c(const char* format, va_list args) {
|
|
if (varused_test) {
|
|
varused = 1;
|
|
return;
|
|
}
|
|
crc_vprintf(format, args);
|
|
}
|
|
|
|
/**
|
|
Print a formatted line to the source file.
|
|
\param[in] format printf-style formatting text, followed by a vararg list
|
|
*/
|
|
void Code_Writer::write_c(const char* format,...) {
|
|
va_list args;
|
|
va_start(args, format);
|
|
vwrite_c(format, args);
|
|
va_end(args);
|
|
}
|
|
|
|
/**
|
|
Write code (c) of size (n) to C file, with optional comment (com) w/o trailing space.
|
|
if the code line does not end in a ';' or '}', a ';' will be added.
|
|
\param[in] indent indentation string for all lines
|
|
\param[in] n number of bytes in code line
|
|
\param[in] c line of code
|
|
\param[in] com optional commentary
|
|
*/
|
|
void Code_Writer::write_cc(const char *indent, int n, const char *c, const char *com) {
|
|
write_c("%s%.*s", indent, n, c);
|
|
char cc = c[n-1];
|
|
if (cc!='}' && cc!=';')
|
|
write_c(";");
|
|
if (*com)
|
|
write_c(" %s", com);
|
|
write_c("\n");
|
|
}
|
|
|
|
/**
|
|
Print a formatted line to the header file.
|
|
\param[in] format printf-style formatting text, followed by a vararg list
|
|
*/
|
|
void Code_Writer::write_h(const char* format,...) {
|
|
if (varused_test) return;
|
|
va_list args;
|
|
va_start(args, format);
|
|
vfprintf(header_file, format, args);
|
|
va_end(args);
|
|
}
|
|
|
|
/**
|
|
Write code (c) of size (n) to H file, with optional comment (com) w/o trailing space.
|
|
if the code line does not end in a ';' or '}', a ';' will be added.
|
|
\param[in] indent indentation string for all lines
|
|
\param[in] n number of bytes in code line
|
|
\param[in] c line of code
|
|
\param[in] com optional commentary
|
|
*/
|
|
void Code_Writer::write_hc(const char *indent, int n, const char* c, const char *com) {
|
|
write_h("%s%.*s", indent, n, c);
|
|
char cc = c[n-1];
|
|
if (cc!='}' && cc!=';')
|
|
write_h(";");
|
|
if (*com)
|
|
write_h(" %s", com);
|
|
write_h("\n");
|
|
}
|
|
|
|
/**
|
|
Write one or more lines of code, indenting each one of them.
|
|
\param[in] textlines one or more lines of text, separated by \\n
|
|
\param[in] inIndent increment indentation by this amount
|
|
\param[in] inTrailWith append this character if the last line did not end with
|
|
a newline, usually 0 or newline.
|
|
*/
|
|
void Code_Writer::write_c_indented(const char *textlines, int inIndent, char inTrailWith) {
|
|
if (textlines) {
|
|
indentation += inIndent;
|
|
for (;;) {
|
|
int line_len;
|
|
const char *newline = strchr(textlines, '\n');
|
|
if (newline)
|
|
line_len = (int)(newline-textlines);
|
|
else
|
|
line_len = (int)strlen(textlines);
|
|
if (textlines[0]=='\n') {
|
|
// avoid trailing spaces
|
|
} else if (textlines[0]=='#') {
|
|
// don't indent preprocessor statments starting with '#'
|
|
write_c("%.*s", line_len, textlines);
|
|
} else {
|
|
// indent all other text lines
|
|
write_c("%s%.*s", indent(), line_len, textlines);
|
|
}
|
|
if (newline) {
|
|
write_c("\n");
|
|
} else {
|
|
if (inTrailWith)
|
|
write_c("%c", inTrailWith);
|
|
break;
|
|
}
|
|
textlines = newline+1;
|
|
}
|
|
indentation -= inIndent;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Return true if the type would be the member of a class.
|
|
Some types are treated differently if they are inside class. Especially within
|
|
a Widget Class, children that are widgets are written as part of the
|
|
constructor whereas functions, declarations, and inline data are seen as
|
|
members of the class itself.
|
|
*/
|
|
bool is_class_member(Node *t) {
|
|
return t->is_a(Type::Function)
|
|
|| t->is_a(Type::Decl)
|
|
|| t->is_a(Type::Data);
|
|
// || t->is_a(Type::Class) // FLUID can't handle a class inside a class
|
|
// || t->is_a(Type::Widget_Class)
|
|
// || t->is_a(Type::DeclBlock) // Declaration blocks are generally not handled well
|
|
}
|
|
|
|
/**
|
|
Return true, if this is a comment, and if it is followed by a class member.
|
|
This must only be called if q is inside a widget class.
|
|
Widget classes can have widgets and members (functions/methods, declarations,
|
|
etc.) intermixed.
|
|
\param[in] q should be a comment type
|
|
\return true if this comment is followed by a class member
|
|
\return false if it is followed by a widget or code
|
|
\see is_class_member(Node *t)
|
|
*/
|
|
bool is_comment_before_class_member(Node *q) {
|
|
if (q->is_a(Type::Comment) && q->next && q->next->level==q->level) {
|
|
if (q->next->is_a(Type::Comment))
|
|
return is_comment_before_class_member(q->next);
|
|
if (is_class_member(q->next))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
Recursively write static code and declarations
|
|
\param[in] p write this type and all its children
|
|
\return pointer to the next sibling
|
|
*/
|
|
Node* Code_Writer::write_static(Node* p) {
|
|
if (write_codeview) p->header_static_start = (int)ftell(header_file);
|
|
if (write_codeview) p->code_static_start = (int)ftell(code_file);
|
|
p->write_static(*this);
|
|
if (write_codeview) p->code_static_end = (int)ftell(code_file);
|
|
if (write_codeview) p->header_static_end = (int)ftell(header_file);
|
|
|
|
Node* q;
|
|
for (q = p->next; q && q->level > p->level;) {
|
|
q = write_static(q);
|
|
}
|
|
|
|
p->write_static_after(*this);
|
|
|
|
return q;
|
|
}
|
|
|
|
/**
|
|
Recursively write code, putting children between the two parts of the parent code.
|
|
\param[in] p write this node and all its children
|
|
\return pointer to the next sibling
|
|
*/
|
|
Node* Code_Writer::write_code(Node* p) {
|
|
// write all code that comes before the children code
|
|
// (but don't write the last comment until the very end)
|
|
if (!(p==Fluid.proj.tree.last && p->is_a(Type::Comment))) {
|
|
if (write_codeview) p->code1_start = (int)ftell(code_file);
|
|
if (write_codeview) p->header1_start = (int)ftell(header_file);
|
|
p->write_code1(*this);
|
|
if (write_codeview) p->code1_end = (int)ftell(code_file);
|
|
if (write_codeview) p->header1_end = (int)ftell(header_file);
|
|
}
|
|
// recursively write the code of all children
|
|
Node* q;
|
|
if (p->is_widget() && p->is_class()) {
|
|
// Handle widget classes specially
|
|
for (q = p->next; q && q->level > p->level;) {
|
|
// note: maybe declaration blocks should be handled like comments in the context
|
|
if (!is_class_member(q) && !is_comment_before_class_member(q)) {
|
|
q = write_code(q);
|
|
} else {
|
|
int level = q->level;
|
|
do {
|
|
q = q->next;
|
|
} while (q && q->level > level);
|
|
}
|
|
}
|
|
|
|
// write all code that come after the children
|
|
if (write_codeview) p->code2_start = (int)ftell(code_file);
|
|
if (write_codeview) p->header2_start = (int)ftell(header_file);
|
|
p->write_code2(*this);
|
|
if (write_codeview) p->code2_end = (int)ftell(code_file);
|
|
if (write_codeview) p->header2_end = (int)ftell(header_file);
|
|
|
|
for (q = p->next; q && q->level > p->level;) {
|
|
if (is_class_member(q) || is_comment_before_class_member(q)) {
|
|
q = write_code(q);
|
|
} else {
|
|
int level = q->level;
|
|
do {
|
|
q = q->next;
|
|
} while (q && q->level > level);
|
|
}
|
|
}
|
|
|
|
write_h("};\n");
|
|
current_widget_class = nullptr;
|
|
} else {
|
|
for (q = p->next; q && q->level > p->level;) q = write_code(q);
|
|
// write all code that come after the children
|
|
if (write_codeview) p->code2_start = (int)ftell(code_file);
|
|
if (write_codeview) p->header2_start = (int)ftell(header_file);
|
|
p->write_code2(*this);
|
|
if (write_codeview) p->code2_end = (int)ftell(code_file);
|
|
if (write_codeview) p->header2_end = (int)ftell(header_file);
|
|
}
|
|
return q;
|
|
}
|
|
|
|
/**
|
|
Write the source and header files for the current design.
|
|
|
|
If the files already exist, they will be overwritten.
|
|
|
|
\note There is no true error checking here.
|
|
|
|
\param[in] s filename of source code file
|
|
\param[in] t filename of the header file
|
|
\return 0 if the operation failed, 1 if it was successful
|
|
*/
|
|
int Code_Writer::write_code(const char *s, const char *t, bool to_codeview) {
|
|
write_codeview = to_codeview;
|
|
unique_id_list.clear();
|
|
indentation = 0;
|
|
current_class = nullptr;
|
|
current_widget_class = nullptr;
|
|
if (!s) code_file = stdout;
|
|
else {
|
|
FILE *f = fl_fopen(s, "wb");
|
|
if (!f) return 0;
|
|
code_file = f;
|
|
}
|
|
if (!t) header_file = stdout;
|
|
else {
|
|
FILE *f = fl_fopen(t, "wb");
|
|
if (!f) {fclose(code_file); return 0;}
|
|
header_file = f;
|
|
}
|
|
// Remember the last code file location for MergeBack
|
|
if (s && proj_.write_mergeback_data && !to_codeview) {
|
|
std::string filename = proj_.projectfile_path() + proj_.projectfile_name();
|
|
int i, n = (int)filename.size();
|
|
for (i=0; i<n; i++) if (filename[i]=='\\') filename[i] = '/';
|
|
Fl_Preferences build_records(Fl_Preferences::USER_L, "fltk.org", "fluid-build");
|
|
Fl_Preferences path(build_records, filename.c_str());
|
|
path.set("code", s);
|
|
}
|
|
// if the first entry in the Type tree is a comment, then it is probably
|
|
// a copyright notice. We print that before anything else in the file!
|
|
Node* first_node = Fluid.proj.tree.first;
|
|
if (first_node && first_node->is_a(Type::Comment)) {
|
|
if (write_codeview) {
|
|
first_node->code1_start = first_node->code2_start = (int)ftell(code_file);
|
|
first_node->header1_start = first_node->header2_start = (int)ftell(header_file);
|
|
}
|
|
// it is ok to write non-recursive code here, because comments have no children or code2 blocks
|
|
first_node->write_code1(*this);
|
|
if (write_codeview) {
|
|
first_node->code1_end = first_node->code2_end = (int)ftell(code_file);
|
|
first_node->header1_end = first_node->header2_end = (int)ftell(header_file);
|
|
}
|
|
first_node = first_node->next;
|
|
}
|
|
|
|
const char *hdr = "\
|
|
// generated by Fast Light User Interface Designer (fluid) version %.4f\n\n";
|
|
fprintf(header_file, hdr, FL_VERSION);
|
|
crc_printf(hdr, FL_VERSION);
|
|
{
|
|
// Creating the include guard is more involved than it seems at first glance.
|
|
// The include guard is deduced from header filename. However, if the
|
|
// filename contains unicode characters, they need to be encoded using
|
|
// \Uxxxxxxxx or \\uxxxx encoding to form a valid macro identifier.
|
|
//
|
|
// But that approach is not portable. Windows does not normalize Unicode
|
|
// (ö is the letter \u00F6). macOS normalizes to NFD (ö is \u006F\u0308,
|
|
// o followed by a Combining Diaresis ¨).
|
|
//
|
|
// To make the include guard consistent across l=platforms, it can be
|
|
// explicitly set by the user in the Project Settings.
|
|
std::string macro_name_str = proj_.include_guard;
|
|
if (macro_name_str.empty()) {
|
|
std::ostringstream macro_name;
|
|
std::string header_name;
|
|
const char* a = nullptr;
|
|
if (write_codeview) {
|
|
header_name = proj_.headerfile_name();
|
|
a = header_name.c_str();
|
|
} else {
|
|
a = fl_filename_name(t);
|
|
}
|
|
const char* b = a + strlen(a);
|
|
int len = 0;
|
|
unsigned ucs = fl_utf8decode(a, b, &len);
|
|
if ((ucs > 127) || (!isalpha(ucs) && (ucs != '_')))
|
|
macro_name << '_';
|
|
while (a < b) {
|
|
ucs = fl_utf8decode(a, b, &len);
|
|
if (ucs > 0x0000ffff) { // large unicode character
|
|
macro_name << "\\U" << std::setw(8) << std::setfill('0') << std::hex << ucs;
|
|
} else if (ucs > 127) { // small unicode character or not an ASCI letter or digit
|
|
macro_name << "\\u" << std::setw(4) << std::setfill('0') << std::hex << ucs;
|
|
} else if (!isalnum(ucs)) {
|
|
macro_name << '_';
|
|
} else {
|
|
macro_name << (char)ucs;
|
|
}
|
|
a += len;
|
|
}
|
|
macro_name_str = macro_name.str();
|
|
}
|
|
fprintf(header_file, "#ifndef %s\n", macro_name_str.c_str());
|
|
fprintf(header_file, "#define %s\n", macro_name_str.c_str());
|
|
}
|
|
|
|
if (proj_.avoid_early_includes==0) {
|
|
write_h_once("#include <FL/Fl.H>");
|
|
}
|
|
if (t && proj_.include_H_from_C) {
|
|
if (to_codeview) {
|
|
write_c("#include \"CodeView.h\"\n");
|
|
} else if (proj_.header_file_name[0] == '.' && strchr(proj_.header_file_name.c_str(), '/') == nullptr) {
|
|
write_c("#include \"%s\"\n", fl_filename_name(t));
|
|
} else {
|
|
write_c("#include \"%s\"\n", proj_.header_file_name.c_str());
|
|
}
|
|
}
|
|
std::string loc_include, loc_conditional;
|
|
if (proj_.i18n.type==fld::I18n_Type::GNU) {
|
|
loc_include = proj_.i18n.gnu_include;
|
|
loc_conditional = proj_.i18n.gnu_conditional;
|
|
} else {
|
|
loc_include = proj_.i18n.posix_include;
|
|
loc_conditional = proj_.i18n.posix_conditional;
|
|
}
|
|
if ((proj_.i18n.type != fld::I18n_Type::NONE) && !loc_include.empty()) {
|
|
int conditional = !loc_conditional.empty();
|
|
if (conditional) {
|
|
write_c("#ifdef %s\n", loc_conditional.c_str());
|
|
indentation++;
|
|
}
|
|
if (loc_include[0] != '<' && loc_include[0] != '\"')
|
|
write_c("#%sinclude \"%s\"\n", indent(), loc_include.c_str());
|
|
else
|
|
write_c("#%sinclude %s\n", indent(), loc_include.c_str());
|
|
if (proj_.i18n.type == fld::I18n_Type::POSIX) {
|
|
if (!proj_.i18n.posix_file.empty()) {
|
|
write_c("extern nl_catd %s;\n", proj_.i18n.posix_file.c_str());
|
|
} else {
|
|
write_c("// Initialize I18N stuff now for menus...\n");
|
|
write_c("#%sinclude <locale.h>\n", indent());
|
|
write_c("static char *_locale = setlocale(LC_MESSAGES, \"\");\n");
|
|
write_c("static nl_catd _catalog = catopen(\"%s\", 0);\n", proj_.basename().c_str());
|
|
}
|
|
}
|
|
if (conditional) {
|
|
write_c("#else\n");
|
|
if (proj_.i18n.type == fld::I18n_Type::GNU) {
|
|
if (!proj_.i18n.gnu_function.empty()) {
|
|
write_c("#%sifndef %s\n", indent(), proj_.i18n.gnu_function.c_str());
|
|
write_c("#%sdefine %s(text) text\n", indent_plus(1), proj_.i18n.gnu_function.c_str());
|
|
write_c("#%sendif\n", indent());
|
|
}
|
|
}
|
|
if (proj_.i18n.type == fld::I18n_Type::POSIX) {
|
|
write_c("#%sifndef catgets\n", indent());
|
|
write_c("#%sdefine catgets(catalog, set, msgid, text) text\n", indent_plus(1));
|
|
write_c("#%sendif\n", indent());
|
|
}
|
|
indentation--;
|
|
write_c("#endif\n");
|
|
}
|
|
if (proj_.i18n.type == fld::I18n_Type::GNU && proj_.i18n.gnu_static_function[0]) {
|
|
write_c("#ifndef %s\n", proj_.i18n.gnu_static_function.c_str());
|
|
write_c("#%sdefine %s(text) text\n", indent_plus(1), proj_.i18n.gnu_static_function.c_str());
|
|
write_c("#endif\n");
|
|
}
|
|
}
|
|
for (Node* p = first_node; p;) {
|
|
// write all static data for this & all children first
|
|
write_static(p);
|
|
// then write the nested code:
|
|
p = write_code(p);
|
|
}
|
|
|
|
if (!s) return 1;
|
|
|
|
fprintf(header_file, "#endif\n");
|
|
|
|
Node* last_node = Fluid.proj.tree.last;
|
|
if (last_node && (last_node != Fluid.proj.tree.first) && last_node->is_a(Type::Comment)) {
|
|
if (write_codeview) {
|
|
last_node->code1_start = last_node->code2_start = (int)ftell(code_file);
|
|
last_node->header1_start = last_node->header2_start = (int)ftell(header_file);
|
|
}
|
|
last_node->write_code1(*this);
|
|
if (write_codeview) {
|
|
last_node->code1_end = last_node->code2_end = (int)ftell(code_file);
|
|
last_node->header1_end = last_node->header2_end = (int)ftell(header_file);
|
|
}
|
|
}
|
|
int x = 0, y = 0;
|
|
|
|
if (code_file != stdout)
|
|
x = fclose(code_file);
|
|
code_file = nullptr;
|
|
if (header_file != stdout)
|
|
y = fclose(header_file);
|
|
header_file = nullptr;
|
|
return x >= 0 && y >= 0;
|
|
}
|
|
|
|
|
|
/**
|
|
Write the public/private/protected keywords inside the class.
|
|
This avoids repeating these words if the mode is already set.
|
|
\param[in] state 0 for private, 1 for public, 2 for protected
|
|
*/
|
|
void Code_Writer::write_public(int state) {
|
|
if (!current_class && !current_widget_class) return;
|
|
if (current_class && current_class->write_public_state == state) return;
|
|
if (current_widget_class && current_widget_class->write_public_state == state) return;
|
|
if (current_class) current_class->write_public_state = state;
|
|
if (current_widget_class) current_widget_class->write_public_state = state;
|
|
switch (state) {
|
|
case 0: write_h("private:\n"); break;
|
|
case 1: write_h("public:\n"); break;
|
|
case 2: write_h("protected:\n"); break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Create and initialize a new C++ source code writer.
|
|
*/
|
|
Code_Writer::Code_Writer(Project &proj)
|
|
: proj_ { proj }
|
|
{
|
|
block_crc_ = crc32(0, nullptr, 0);
|
|
}
|
|
|
|
/**
|
|
Release all resources.
|
|
*/
|
|
Code_Writer::~Code_Writer()
|
|
{
|
|
if (block_buffer_) ::free(block_buffer_);
|
|
}
|
|
|
|
/**
|
|
Write a MergeBack tag as a separate line of C++ comment.
|
|
The tag contains information about the type of tag that we are writing, a
|
|
link back to the type using its unique id, and the CRC of all code written
|
|
after the previous tag up to this point.
|
|
\param[in] type FD_TAG_GENERIC, FD_TAG_CODE, FD_TAG_MENU_CALLBACK, or FD_TAG_WIDGET_CALLBACK
|
|
\param[in] uid the unique id of the current type
|
|
*/
|
|
void Code_Writer::tag(proj::Mergeback::Tag prev_type, proj::Mergeback::Tag next_type, unsigned short uid) {
|
|
if (proj_.write_mergeback_data) {
|
|
Mergeback::print_tag(code_file, prev_type, next_type, uid, (uint32_t)block_crc_);
|
|
}
|
|
block_crc_ = crc32(0, nullptr, 0);
|
|
}
|
|
|
|
/**
|
|
Static function to calculate the CRC32 of a block of C source code.
|
|
Calculation of the CRC ignores leading whitespace in a line and all linefeed
|
|
characters ('\\r').
|
|
\param[in] data a pointer to the data block
|
|
\param[in] n the size of the data in bytes, or -1 to use strlen()
|
|
\param[in] in_crc add to this CRC, 0 by default to start a new block
|
|
\param[inout] inout_line_start optional pointer to flag that determines
|
|
if we are the start of a line, used to find leading whitespace
|
|
\return the new CRC
|
|
*/
|
|
unsigned long Code_Writer::block_crc(const void *data, int n, unsigned long in_crc, bool *inout_line_start) {
|
|
if (!data) return 0;
|
|
if (n==-1) n = (int)strlen((const char*)data);
|
|
bool line_start = true;
|
|
if (inout_line_start) line_start = *inout_line_start;
|
|
const char *s = (const char*)data;
|
|
for ( ; n>0; --n, ++s) {
|
|
if (line_start) {
|
|
// don't count leading spaces and tabs in a line
|
|
while (n>0 && *s>0 && isspace(*s)) { s++; n--; }
|
|
if (*s) line_start = false;
|
|
}
|
|
// don't count '\r' that may be introduced by Windows
|
|
if (n>0 && *s=='\r') { s++; n--; }
|
|
if (n>0 && *s=='\n') line_start = true;
|
|
if (n>0) {
|
|
in_crc = crc32(in_crc, (const Bytef*)s, 1);
|
|
}
|
|
}
|
|
if (inout_line_start) *inout_line_start = line_start;
|
|
return in_crc;
|
|
}
|
|
|
|
/** Add the following block of text to the CRC of this class.
|
|
\param[in] data a pointer to the data block
|
|
\param[in] n the size of the data in bytes, or -1 to use strlen()
|
|
*/
|
|
void Code_Writer::crc_add(const void *data, int n) {
|
|
block_crc_ = block_crc(data, n, block_crc_, &block_line_start_);
|
|
}
|
|
|
|
/** Write formatted text to the code file.
|
|
If MergeBack is enabled, the CRC calculation is continued.
|
|
\param[in] format printf style formatting string
|
|
\return see fprintf(FILE *, *const char*, ...)
|
|
*/
|
|
int Code_Writer::crc_printf(const char *format, ...) {
|
|
va_list args;
|
|
va_start(args, format);
|
|
int ret = crc_vprintf(format, args);
|
|
va_end(args);
|
|
return ret;
|
|
}
|
|
|
|
/** Write formatted text to the code file.
|
|
If MergeBack is enabled, the CRC calculation is continued.
|
|
\param[in] format printf style formatting string
|
|
\param[in] args list of arguments
|
|
\return see fprintf(FILE *, *const char*, ...)
|
|
*/
|
|
int Code_Writer::crc_vprintf(const char *format, va_list args) {
|
|
if (proj_.write_mergeback_data) {
|
|
int n = vsnprintf(block_buffer_, block_buffer_size_, format, args);
|
|
if (n > block_buffer_size_) {
|
|
block_buffer_size_ = n + 128;
|
|
if (block_buffer_) ::free(block_buffer_);
|
|
block_buffer_ = (char*)::malloc(block_buffer_size_+1);
|
|
n = vsnprintf(block_buffer_, block_buffer_size_, format, args);
|
|
}
|
|
crc_add(block_buffer_, n);
|
|
return fputs(block_buffer_, code_file);
|
|
} else {
|
|
return vfprintf(code_file, format, args);
|
|
}
|
|
}
|
|
|
|
/** Write some text to the code file.
|
|
If MergeBack is enabled, the CRC calculation is continued.
|
|
\param[in] text any text, no requirements to end in a newline or such
|
|
\return see fputs(const char*, FILE*)
|
|
*/
|
|
int Code_Writer::crc_puts(const char *text) {
|
|
if (proj_.write_mergeback_data) {
|
|
crc_add(text);
|
|
}
|
|
return fputs(text, code_file);
|
|
}
|
|
|
|
/** Write a single ASCII character to the code file.
|
|
If MergeBack is enabled, the CRC calculation is continued.
|
|
\note to write UTF-8 characters, use Code_Writer::crc_puts(const char *text)
|
|
\param[in] c any character between 0 and 127 inclusive
|
|
\return see fputc(int, FILE*)
|
|
*/
|
|
int Code_Writer::crc_putc(int c) {
|
|
if (proj_.write_mergeback_data) {
|
|
uchar uc = (uchar)c;
|
|
crc_add(&uc, 1);
|
|
}
|
|
return fputc(c, code_file);
|
|
}
|
|
|