aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--zathura-note/note.c290
1 files changed, 253 insertions, 37 deletions
diff --git a/zathura-note/note.c b/zathura-note/note.c
index 2d12114..86a25a7 100644
--- a/zathura-note/note.c
+++ b/zathura-note/note.c
@@ -10,6 +10,104 @@ typedef struct {
plist_t metadata_plist;
} note_document_t;
+// Found by reverse engineering
+#define SESSION_OBJECTS_GENERAL_INFO 1
+#define SESSION_OBJECTS_FORMAT_INFO 2
+
+// For debugging/reverse engineering
+#define INDENT 4
+static void plist_dump(plist_t plist, int depth)
+{
+ for (int i = 0; i < depth * INDENT; i++)
+ printf(" ");
+
+ if (PLIST_IS_BOOLEAN(plist)) {
+ unsigned char val;
+ plist_get_bool_val(plist, &val);
+ printf("<bool>%s</bool>\n", val ? "true" : "false");
+ } else if (PLIST_IS_UINT(plist)) {
+ unsigned long val;
+ plist_get_uint_val(plist, &val);
+ printf("<uint>%lu</uint>\n", val);
+ } else if (PLIST_IS_REAL(plist)) {
+ double val;
+ plist_get_real_val(plist, &val);
+ printf("<real>%f</real>\n", val);
+ } else if (PLIST_IS_STRING(plist)) {
+ char *val;
+ plist_get_string_val(plist, &val);
+ printf("<string>%s</string>\n", val);
+ free(val);
+ } else if (PLIST_IS_ARRAY(plist)) {
+ plist_array_iter iter;
+ plist_array_new_iter(plist, &iter);
+ plist_t val;
+ printf("<array>\n");
+ int id = 0;
+ while (1) {
+ plist_array_next_item(plist, iter, &val);
+ if (!val)
+ break;
+
+ for (int i = 0; i < (depth + 1) * INDENT; i++)
+ printf(" ");
+ printf("<array_item id=\"%d\">\n", id++);
+
+ plist_dump(val, depth + 2);
+
+ for (int i = 0; i < (depth + 1) * INDENT; i++)
+ printf(" ");
+ printf("</array_item>\n");
+ }
+ for (int i = 0; i < depth * INDENT; i++)
+ printf(" ");
+ printf("</array>\n");
+ } else if (PLIST_IS_DICT(plist)) {
+ plist_dict_iter iter;
+ plist_dict_new_iter(plist, &iter);
+ char *key = 0;
+ plist_t val;
+ printf("<dict>\n");
+ int id = 0;
+ while (1) {
+ plist_dict_next_item(plist, iter, &key, &val);
+ if (!key || !val)
+ break;
+
+ for (int i = 0; i < (depth + 1) * INDENT; i++)
+ printf(" ");
+ printf("<dict_item key=\"%s\" id=\"%d\">\n", key, id++);
+
+ plist_dump(val, depth + 2);
+ free(key);
+
+ for (int i = 0; i < (depth + 1) * INDENT; i++)
+ printf(" ");
+ printf("</dict_item>\n");
+ }
+ for (int i = 0; i < depth * INDENT; i++)
+ printf(" ");
+ printf("</dict>\n");
+ } else if (PLIST_IS_DATE(plist)) {
+ int sec, usec;
+ plist_get_date_val(plist, &sec, &usec);
+ printf("<date>%d</date>\n", sec); // Since 01/01/2001
+ } else if (PLIST_IS_DATA(plist)) {
+ unsigned long length;
+ const char *val = plist_get_data_ptr(plist, &length);
+ printf("<data length=\"%lu\">%.*s</data>\n", length, (int)length, val);
+ } else if (PLIST_IS_KEY(plist)) {
+ char *val;
+ plist_get_key_val(plist, &val);
+ printf("<key>%s</key>\n", val);
+ free(val);
+ } else if (PLIST_IS_UID(plist)) {
+ unsigned long val;
+ plist_get_uid_val(plist, &val);
+ printf("<uid>%lu</uid>\n", val);
+ }
+}
+
static zathura_error_t plist_load(zip_t *zip, plist_t *plist, const char *root_name,
const char *path)
{
@@ -26,14 +124,129 @@ static zathura_error_t plist_load(zip_t *zip, plist_t *plist, const char *root_n
void *bin = malloc(stat.size);
size_t length = zip_fread(file, bin, stat.size);
- if (length < stat.size)
+ if (length < stat.size) {
fprintf(stderr, "Unexpected size difference\n");
+ free(bin);
+ return ZATHURA_ERROR_INVALID_ARGUMENTS;
+ }
+
+ if (!plist_is_binary(bin, stat.size)) {
+ fprintf(stderr, "Unexpected file format of '%s'\n", path);
+ free(bin);
+ return ZATHURA_ERROR_INVALID_ARGUMENTS;
+ }
plist_from_bin(bin, stat.size, plist);
free(bin);
return ZATHURA_ERROR_OK;
}
+static plist_t plist_session_objects(plist_t session_plist)
+{
+ plist_t objects = plist_dict_get_item(session_plist, "$objects");
+ if (!PLIST_IS_ARRAY(objects)) {
+ fprintf(stderr, "Invalid objects type\n");
+ return 0;
+ }
+ return objects;
+}
+
+static plist_t plist_session_format_information(plist_t session_plist)
+{
+ plist_t objects = plist_session_objects(session_plist);
+ if (!objects)
+ return 0;
+
+ plist_t format = plist_array_get_item(objects, SESSION_OBJECTS_FORMAT_INFO);
+ if (!PLIST_IS_DICT(format)) {
+ fprintf(stderr, "Invalid format information type\n");
+ return 0;
+ }
+ return format;
+}
+
+static plist_t plist_handwriting_overlay(plist_t session_plist)
+{
+ plist_t objects = plist_session_objects(session_plist);
+ if (!objects)
+ return 0;
+
+ plist_t format = plist_array_get_item(objects, SESSION_OBJECTS_FORMAT_INFO);
+ if (!PLIST_IS_DICT(format)) {
+ fprintf(stderr, "Invalid format information\n");
+ return 0;
+ }
+
+ plist_t overlay_pointer = plist_dict_get_item(format, "Handwriting Overlay");
+ if (!PLIST_IS_UID(overlay_pointer)) {
+ fprintf(stderr, "Invalid handwriting overlay pointer\n");
+ return 0;
+ }
+
+ unsigned long index;
+ plist_get_uid_val(overlay_pointer, &index);
+
+ plist_t overlay_info = plist_array_get_item(objects, index);
+ if (!PLIST_IS_DICT(overlay_info)) {
+ fprintf(stderr, "Invalid overlay info item\n");
+ return 0;
+ }
+
+ plist_t spatial_hash = plist_dict_get_item(overlay_info, "SpatialHash");
+ if (!PLIST_IS_UID(spatial_hash)) {
+ fprintf(stderr, "Invalid spatial hash\n");
+ return 0;
+ }
+
+ plist_get_uid_val(spatial_hash, &index);
+
+ plist_t overlay = plist_array_get_item(objects, index);
+ if (!PLIST_IS_DICT(overlay)) {
+ fprintf(stderr, "Invalid handwriting overlay\n");
+ return 0;
+ }
+
+ return overlay;
+}
+
+static float plist_page_width(plist_t session_plist)
+{
+ plist_t objects = plist_session_objects(session_plist);
+ if (!objects)
+ return 0;
+
+ plist_t format = plist_array_get_item(objects, SESSION_OBJECTS_FORMAT_INFO);
+ if (!PLIST_IS_DICT(format)) {
+ fprintf(stderr, "Invalid format information\n");
+ return 0;
+ }
+
+ plist_t reflow_state_pointer = plist_dict_get_item(format, "reflowState");
+ if (!PLIST_IS_UID(reflow_state_pointer)) {
+ fprintf(stderr, "Invalid reflow state pointer\n");
+ return 0;
+ }
+
+ unsigned long index;
+ plist_get_uid_val(reflow_state_pointer, &index);
+
+ plist_t reflow_state = plist_array_get_item(objects, index);
+ if (!PLIST_IS_DICT(reflow_state)) {
+ fprintf(stderr, "Invalid reflow state item\n");
+ return 0;
+ }
+
+ plist_t page_width = plist_dict_get_item(reflow_state, "pageWidthInDocumentCoordsKey");
+ if (!PLIST_IS_REAL(page_width)) {
+ fprintf(stderr, "Invalid page width\n");
+ return 0;
+ }
+
+ double val;
+ plist_get_real_val(page_width, &val);
+ return val;
+}
+
GIRARA_HIDDEN zathura_error_t note_document_open(zathura_document_t *document)
{
zathura_error_t error = ZATHURA_ERROR_OK;
@@ -55,7 +268,7 @@ GIRARA_HIDDEN zathura_error_t note_document_open(zathura_document_t *document)
char *root_name;
zip_stat_t root_stat;
- if (!zip_stat_index(zip, 0, ZIP_FL_NODIR, &root_stat)) {
+ if (!zip_stat_index(zip, 0, 0, &root_stat)) {
int length = strlen(root_stat.name);
root_name = malloc(length + 1);
strcpy(root_name, root_stat.name);
@@ -97,6 +310,9 @@ GIRARA_HIDDEN zathura_error_t note_document_open(zathura_document_t *document)
GIRARA_HIDDEN zathura_error_t note_document_free(zathura_document_t *document, void *data)
{
+ if (!data)
+ return ZATHURA_ERROR_OK;
+
note_document_t *note_document = data;
zip_close(note_document->zip);
return ZATHURA_ERROR_OK;
@@ -104,10 +320,13 @@ GIRARA_HIDDEN zathura_error_t note_document_free(zathura_document_t *document, v
GIRARA_HIDDEN zathura_error_t note_page_init(zathura_page_t *page)
{
- // TODO: Get width dynamagically
- int width = 500;
+ note_document_t *note_document = zathura_document_get_data(zathura_page_get_document(page));
+ float width = plist_page_width(note_document->session_plist);
+ if (width < 1)
+ return ZATHURA_ERROR_NOT_IMPLEMENTED;
+
zathura_page_set_width(page, width);
- zathura_page_set_height(page, (int)((float)width * 1.41));
+ zathura_page_set_height(page, width * 1.41); // Always A4?
return ZATHURA_ERROR_OK;
}
@@ -117,44 +336,41 @@ GIRARA_HIDDEN zathura_error_t note_page_clear(zathura_page_t *page, void *data)
return ZATHURA_ERROR_OK;
}
-// For debugging
-static void plist_dump(plist_t plist, int depth)
-{
- // Debug dump
- plist_dict_iter iter;
- plist_dict_new_iter(plist, &iter);
- char *key = 0;
- plist_t val;
- while (1) {
- plist_dict_next_item(plist, iter, &key, &val);
- if (!key || !val)
- break;
-
- printf("%s\n", key);
- for (int i = 0; i < depth; i++)
- printf(" ");
- }
-}
-
GIRARA_HIDDEN zathura_error_t note_page_render_cairo(zathura_page_t *page, void *data,
cairo_t *cairo, bool printing)
{
- printf("Rendering page %d\n", zathura_page_get_index(page));
if (printing)
return ZATHURA_ERROR_NOT_IMPLEMENTED;
- note_document_t *note_document = data;
- plist_dump(note_document->session_plist, 0);
-
- /* cairo_set_source_rgba(cairo, 0xff, 0, 0, 1); */
- /* cairo_set_line_width(cairo, 1); */
- /* cairo_move_to(cairo, 0, 0); */
- /* cairo_line_to(cairo, 100, 100); */
- /* cairo_rel_line_to(cairo, 0.25, -0.125); */
- /* cairo_arc(cairo, 0.5, 0.5, 0.25 * sqrt(2), -0.25 * M_PI, 0.25 * M_PI); */
- /* cairo_rel_curve_to(cairo, -0.25, -0.125, -0.25, 0.125, -0.5, 0); */
- /* cairo_close_path(cairo); */
- /* cairo_stroke(cairo); */
+ float width = zathura_page_get_width(page);
+ float height = zathura_page_get_height(page);
+
+ note_document_t *note_document = zathura_document_get_data(zathura_page_get_document(page));
+ plist_t overlay = plist_handwriting_overlay(note_document->session_plist);
+ if (!overlay)
+ return ZATHURA_ERROR_NOT_IMPLEMENTED;
+
+ plist_t points_data = plist_dict_get_item(overlay, "curvespoints");
+ if (!PLIST_IS_DATA(points_data))
+ return ZATHURA_ERROR_NOT_IMPLEMENTED;
+
+ unsigned long points_length;
+ const char *points_chars = plist_get_data_ptr(points_data, &points_length);
+ if (!points_chars)
+ return ZATHURA_ERROR_NOT_IMPLEMENTED;
+ float *points = (float *)points_chars;
+
+ cairo_set_source_rgba(cairo, 0xff, 0, 0, 1);
+ cairo_set_line_width(cairo, 1);
+ for (unsigned long i = 0; i < points_length; i += 2) {
+ float x = points[i];
+ float y = points[i + 1];
+ if (x > width || y > height)
+ continue;
+ cairo_line_to(cairo, points[i], points[i + 1]);
+ }
+ cairo_close_path(cairo);
+ cairo_stroke(cairo);
return ZATHURA_ERROR_OK;
}