2.5
[blender-staging.git] / source / blender / editors / gpencil / gpencil.c
1 /**
2  * $Id$
3  *
4  * ***** BEGIN GPL LICENSE BLOCK *****
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software Foundation,
18  * Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
19  *
20  * The Original Code is Copyright (C) 2008, Blender Foundation
21  * This is a new part of Blender
22  *
23  * Contributor(s): Joshua Leung
24  *
25  * ***** END GPL LICENSE BLOCK *****
26  */
27  
28 #include <stdio.h>
29 #include <string.h>
30 #include <stdlib.h>
31 #include <stddef.h>
32 #include <math.h>
33
34 #include "MEM_guardedalloc.h"
35
36 #include "BMF_Api.h"
37
38 #include "IMB_imbuf.h"
39 #include "IMB_imbuf_types.h"
40
41 #include "BLI_arithb.h"
42 #include "BLI_blenlib.h"
43
44 #include "DNA_listBase.h"
45 #include "DNA_armature_types.h"
46 #include "DNA_curve_types.h"
47 #include "DNA_gpencil_types.h"
48 #include "DNA_object_types.h"
49 #include "DNA_scene_types.h"
50 #include "DNA_screen_types.h"
51 #include "DNA_space_types.h"
52 #include "DNA_userdef_types.h"
53 #include "DNA_vec_types.h"
54 #include "DNA_view3d_types.h"
55
56 #include "BKE_armature.h"
57 #include "BKE_blender.h"
58 #include "BKE_context.h"
59 #include "BKE_curve.h"
60 #include "BKE_global.h"
61 #include "BKE_image.h"
62 #include "BKE_library.h"
63 #include "BKE_utildefines.h"
64
65 #include "BIF_gl.h"
66 #include "BIF_glutil.h"
67
68 #include "WM_api.h"
69 #include "WM_types.h"
70
71 #include "UI_view2d.h"
72
73 #include "ED_armature.h"
74 #include "ED_gpencil.h"
75 #include "ED_sequencer.h"
76 #include "ED_view3d.h"
77
78 #include "PIL_time.h"                   /* sleep                                */
79
80 #include "gpencil_intern.h"
81
82 /* XXX */
83 static void BIF_undo_push() {}
84 static void error() {}
85 static int pupmenu() {return 0;}
86 static void add_object_draw() {}
87 static int get_activedevice() {return 0;}
88 #define L_MOUSE 0
89 #define R_MOUSE 0
90
91 /* XXX */
92
93 /* ************************************************** */
94 /* GENERAL STUFF */
95
96 /* --------- Memory Management ------------ */
97
98 /* Free strokes belonging to a gp-frame */
99 void free_gpencil_strokes (bGPDframe *gpf)
100 {
101         bGPDstroke *gps, *gpsn;
102         
103         /* error checking */
104         if (gpf == NULL) return;
105         
106         /* free strokes */
107         for (gps= gpf->strokes.first; gps; gps= gpsn) {
108                 gpsn= gps->next;
109                 
110                 /* free stroke memory arrays, then stroke itself */
111                 if (gps->points) MEM_freeN(gps->points);
112                 BLI_freelinkN(&gpf->strokes, gps);
113         }
114 }
115
116 /* Free all of a gp-layer's frames */
117 void free_gpencil_frames (bGPDlayer *gpl)
118 {
119         bGPDframe *gpf, *gpfn;
120         
121         /* error checking */
122         if (gpl == NULL) return;
123         
124         /* free frames */
125         for (gpf= gpl->frames.first; gpf; gpf= gpfn) {
126                 gpfn= gpf->next;
127                 
128                 /* free strokes and their associated memory */
129                 free_gpencil_strokes(gpf);
130                 BLI_freelinkN(&gpl->frames, gpf);
131         }
132 }
133
134 /* Free all of the gp-layers for a viewport (list should be &gpd->layers or so) */
135 void free_gpencil_layers (ListBase *list) 
136 {
137         bGPDlayer *gpl, *gpln;
138         
139         /* error checking */
140         if (list == NULL) return;
141         
142         /* delete layers*/
143         for (gpl= list->first; gpl; gpl= gpln) {
144                 gpln= gpl->next;
145                 
146                 /* free layers and their data */
147                 free_gpencil_frames(gpl);
148                 BLI_freelinkN(list, gpl);
149         }
150 }
151
152 /* Free gp-data and all it's related data */
153 void free_gpencil_data (bGPdata *gpd)
154 {
155         /* free layers then data itself */
156         free_gpencil_layers(&gpd->layers);
157         MEM_freeN(gpd);
158 }
159
160 /* -------- Container Creation ---------- */
161
162 /* add a new gp-frame to the given layer */
163 bGPDframe *gpencil_frame_addnew (bGPDlayer *gpl, int cframe)
164 {
165         bGPDframe *gpf, *gf;
166         short state=0;
167         
168         /* error checking */
169         if ((gpl == NULL) || (cframe <= 0))
170                 return NULL;
171                 
172         /* allocate memory for this frame */
173         gpf= MEM_callocN(sizeof(bGPDframe), "bGPDframe");
174         gpf->framenum= cframe;
175         
176         /* find appropriate place to add frame */
177         if (gpl->frames.first) {
178                 for (gf= gpl->frames.first; gf; gf= gf->next) {
179                         /* check if frame matches one that is supposed to be added */
180                         if (gf->framenum == cframe) {
181                                 state= -1;
182                                 break;
183                         }
184                         
185                         /* if current frame has already exceeded the frame to add, add before */
186                         if (gf->framenum > cframe) {
187                                 BLI_insertlinkbefore(&gpl->frames, gf, gpf);
188                                 state= 1;
189                                 break;
190                         }
191                 }
192         }
193         
194         /* check whether frame was added successfully */
195         if (state == -1) {
196                 MEM_freeN(gpf);
197                 printf("Error: frame (%d) existed already for this layer \n", cframe);
198         }
199         else if (state == 0) {
200                 /* add to end then! */
201                 BLI_addtail(&gpl->frames, gpf);
202         }
203         
204         /* return frame */
205         return gpf;
206 }
207
208 /* add a new gp-layer and make it the active layer */
209 bGPDlayer *gpencil_layer_addnew (bGPdata *gpd)
210 {
211         bGPDlayer *gpl;
212         
213         /* check that list is ok */
214         if (gpd == NULL)
215                 return NULL;
216                 
217         /* allocate memory for frame and add to end of list */
218         gpl= MEM_callocN(sizeof(bGPDlayer), "bGPDlayer");
219         
220         /* add to datablock */
221         BLI_addtail(&gpd->layers, gpl);
222         
223         /* set basic settings */
224         gpl->color[3]= 0.9f;
225         gpl->thickness = 3;
226         
227         /* auto-name */
228         sprintf(gpl->info, "GP_Layer");
229         BLI_uniquename(&gpd->layers, gpl, "GP_Layer", offsetof(bGPDlayer, info[0]), 128);
230         
231         /* make this one the active one */
232         gpencil_layer_setactive(gpd, gpl);
233         
234         /* return layer */
235         return gpl;
236 }
237
238 /* add a new gp-datablock */
239 bGPdata *gpencil_data_addnew (void)
240 {
241         bGPdata *gpd;
242         
243         /* allocate memory for a new block */
244         gpd= MEM_callocN(sizeof(bGPdata), "GreasePencilData");
245         
246         /* initial settings */
247         gpd->flag = (GP_DATA_DISPINFO|GP_DATA_EXPAND);
248         
249         return gpd;
250 }
251
252 /* -------- Data Duplication ---------- */
253
254 /* make a copy of a given gpencil frame */
255 bGPDframe *gpencil_frame_duplicate (bGPDframe *src)
256 {
257         bGPDstroke *gps, *gpsd;
258         bGPDframe *dst;
259         
260         /* error checking */
261         if (src == NULL)
262                 return NULL;
263                 
264         /* make a copy of the source frame */
265         dst= MEM_dupallocN(src);
266         dst->prev= dst->next= NULL;
267         
268         /* copy strokes */
269         dst->strokes.first = dst->strokes.last= NULL;
270         for (gps= src->strokes.first; gps; gps= gps->next) {
271                 /* make copy of source stroke, then adjust pointer to points too */
272                 gpsd= MEM_dupallocN(gps);
273                 gpsd->points= MEM_dupallocN(gps->points);
274                 
275                 BLI_addtail(&dst->strokes, gpsd);
276         }
277         
278         /* return new frame */
279         return dst;
280 }
281
282 /* make a copy of a given gpencil layer */
283 bGPDlayer *gpencil_layer_duplicate (bGPDlayer *src)
284 {
285         bGPDframe *gpf, *gpfd;
286         bGPDlayer *dst;
287         
288         /* error checking */
289         if (src == NULL)
290                 return NULL;
291                 
292         /* make a copy of source layer */
293         dst= MEM_dupallocN(src);
294         dst->prev= dst->next= NULL;
295         
296         /* copy frames */
297         dst->frames.first= dst->frames.last= NULL;
298         for (gpf= src->frames.first; gpf; gpf= gpf->next) {
299                 /* make a copy of source frame */
300                 gpfd= gpencil_frame_duplicate(gpf);
301                 BLI_addtail(&dst->frames, gpfd);
302                 
303                 /* if source frame was the current layer's 'active' frame, reassign that too */
304                 if (gpf == dst->actframe)
305                         dst->actframe= gpfd;
306         }
307         
308         /* return new layer */
309         return dst;
310 }
311
312 /* make a copy of a given gpencil datablock */
313 bGPdata *gpencil_data_duplicate (bGPdata *src)
314 {
315         bGPDlayer *gpl, *gpld;
316         bGPdata *dst;
317         
318         /* error checking */
319         if (src == NULL)
320                 return NULL;
321         
322         /* make a copy of the base-data */
323         dst= MEM_dupallocN(src);
324         
325         /* copy layers */
326         dst->layers.first= dst->layers.last= NULL;
327         for (gpl= src->layers.first; gpl; gpl= gpl->next) {
328                 /* make a copy of source layer and its data */
329                 gpld= gpencil_layer_duplicate(gpl);
330                 BLI_addtail(&dst->layers, gpld);
331         }
332         
333         /* return new */
334         return dst;
335 }
336
337 /* ----------- GP-Datablock API ------------- */
338
339 /* get the appropriate bGPdata from the active/given context */
340 // XXX region or region data?
341 bGPdata *gpencil_data_getactive (ScrArea *sa)
342 {
343         ScrArea *curarea= NULL; // XXX
344         
345         /* error checking */
346         if ((sa == NULL) && (curarea == NULL))
347                 return NULL;
348         if (sa == NULL)
349                 sa= curarea;
350                 
351         /* handle depending on spacetype */
352         switch (sa->spacetype) {
353                 case SPACE_VIEW3D:
354                 {
355                         View3D *v3d= sa->spacedata.first;
356                         return v3d->gpd;
357                 }
358                         break;
359                 case SPACE_NODE:
360                 {
361                         SpaceNode *snode= sa->spacedata.first;
362                         return snode->gpd;
363                 }
364                         break;
365                 case SPACE_SEQ:
366                 {
367                         SpaceSeq *sseq= sa->spacedata.first;
368                         
369                         /* only applicable for image modes */
370                         if (sseq->mainb)
371                                 return sseq->gpd;
372                 }
373                         break;
374                 case SPACE_IMAGE:
375                 {
376                         SpaceImage *sima= sa->spacedata.first;
377                         return sima->gpd;
378                 }
379                         break;
380         }
381         
382         /* nothing found */
383         return NULL;
384 }
385
386 /* set bGPdata for the active/given context, and return success/fail */
387 short gpencil_data_setactive (ScrArea *sa, bGPdata *gpd)
388 {
389         ScrArea *curarea= NULL; // XXX
390         
391         /* error checking */
392         if ((sa == NULL) && (curarea == NULL))
393                 return 0;
394         if (gpd == NULL)
395                 return 0;
396         if (sa == NULL)
397                 sa= curarea;
398         
399         /* handle depending on spacetype */
400         // TODO: someday we should have multi-user data, so no need to loose old data
401         switch (sa->spacetype) {
402                 case SPACE_VIEW3D:
403                 {
404                         View3D *v3d= sa->spacedata.first;
405                         
406                         /* free the existing block */
407                         if (v3d->gpd)
408                                 free_gpencil_data(v3d->gpd);
409                         v3d->gpd= gpd;
410                         
411                         return 1;
412                 }
413                         break;
414                 case SPACE_NODE:
415                 {
416                         SpaceNode *snode= sa->spacedata.first;
417                         
418                         /* free the existing block */
419                         if (snode->gpd)
420                                 free_gpencil_data(snode->gpd);
421                         snode->gpd= gpd;
422                         
423                         /* set special settings */
424                         gpd->flag |= GP_DATA_VIEWALIGN;
425                         
426                         return 1;
427                 }
428                         break;
429                 case SPACE_SEQ:
430                 {
431                         SpaceSeq *sseq= sa->spacedata.first;
432                         
433                         /* only applicable if right mode */
434                         if (sseq->mainb) {
435                                 /* free the existing block */
436                                 if (sseq->gpd)
437                                         free_gpencil_data(sseq->gpd);
438                                 sseq->gpd= gpd;
439                                 
440                                 return 1;
441                         }
442                 }
443                         break;
444                 case SPACE_IMAGE:
445                 {
446                         SpaceImage *sima= sa->spacedata.first;
447                         
448                         if (sima->gpd)
449                                 free_gpencil_data(sima->gpd);
450                         sima->gpd= gpd;
451                         
452                         return 1;
453                 }
454                         break;
455         }
456         
457         /* failed to add */
458         return 0;
459 }
460
461 /* return the ScrArea that has the given GP-datablock
462  *      - assumes that only searching in current screen
463  *      - is based on GP-datablocks only being able to 
464  *        exist for one area at a time (i.e. not multiuser)
465  */
466 ScrArea *gpencil_data_findowner (bGPdata *gpd)
467 {
468         bScreen *curscreen= NULL; // XXX
469         ScrArea *sa;
470         
471         /* error checking */
472         if (gpd == NULL)
473                 return NULL;
474                 
475         /* loop over all scrareas for current screen, and check if that area has this gpd */
476         for (sa= curscreen->areabase.first; sa; sa= sa->next) {
477                 /* use get-active func to see if match */
478                 if (gpencil_data_getactive(sa) == gpd)
479                         return sa;
480         }
481         
482         /* not found */
483         return NULL;
484 }
485
486 /* -------- GP-Frame API ---------- */
487
488 /* delete the last stroke of the given frame */
489 void gpencil_frame_delete_laststroke (bGPDframe *gpf)
490 {
491         bGPDstroke *gps= (gpf) ? gpf->strokes.last : NULL;
492         
493         /* error checking */
494         if (ELEM(NULL, gpf, gps))
495                 return;
496         
497         /* free the stroke and its data */
498         MEM_freeN(gps->points);
499         BLI_freelinkN(&gpf->strokes, gps);
500 }
501
502 /* -------- GP-Layer API ---------- */
503
504 /* get the appropriate gp-frame from a given layer
505  *      - this sets the layer's actframe var (if allowed to)
506  *      - extension beyond range (if first gp-frame is after all frame in interest and cannot add)
507  */
508 bGPDframe *gpencil_layer_getframe (bGPDlayer *gpl, int cframe, short addnew)
509 {
510         bGPDframe *gpf = NULL;
511         short found = 0;
512         
513         /* error checking */
514         if (gpl == NULL) return NULL;
515         if (cframe <= 0) cframe = 1;
516         
517         /* check if there is already an active frame */
518         if (gpl->actframe) {
519                 gpf= gpl->actframe;
520                 
521                 /* do not allow any changes to layer's active frame if layer is locked */
522                 if (gpl->flag & GP_LAYER_LOCKED)
523                         return gpf;
524                 /* do not allow any changes to actframe if frame has painting tag attached to it */
525                 if (gpf->flag & GP_FRAME_PAINT) 
526                         return gpf;
527                 
528                 /* try to find matching frame */
529                 if (gpf->framenum < cframe) {
530                         for (; gpf; gpf= gpf->next) {
531                                 if (gpf->framenum == cframe) {
532                                         found= 1;
533                                         break;
534                                 }
535                                 else if ((gpf->next) && (gpf->next->framenum > cframe)) {
536                                         found= 1;
537                                         break;
538                                 }
539                         }
540                         
541                         /* set the appropriate frame */
542                         if (addnew) {
543                                 if ((found) && (gpf->framenum == cframe))
544                                         gpl->actframe= gpf;
545                                 else
546                                         gpl->actframe= gpencil_frame_addnew(gpl, cframe);
547                         }
548                         else if (found)
549                                 gpl->actframe= gpf;
550                         else
551                                 gpl->actframe= gpl->frames.last;
552                 }
553                 else {
554                         for (; gpf; gpf= gpf->prev) {
555                                 if (gpf->framenum <= cframe) {
556                                         found= 1;
557                                         break;
558                                 }
559                         }
560                         
561                         /* set the appropriate frame */
562                         if (addnew) {
563                                 if ((found) && (gpf->framenum == cframe))
564                                         gpl->actframe= gpf;
565                                 else
566                                         gpl->actframe= gpencil_frame_addnew(gpl, cframe);
567                         }
568                         else if (found)
569                                 gpl->actframe= gpf;
570                         else
571                                 gpl->actframe= gpl->frames.first;
572                 }
573         }
574         else if (gpl->frames.first) {
575                 /* check which of the ends to start checking from */
576                 const int first= ((bGPDframe *)(gpl->frames.first))->framenum;
577                 const int last= ((bGPDframe *)(gpl->frames.last))->framenum;
578                 
579                 if (abs(cframe-first) > abs(cframe-last)) {
580                         /* find gp-frame which is less than or equal to cframe */
581                         for (gpf= gpl->frames.last; gpf; gpf= gpf->prev) {
582                                 if (gpf->framenum <= cframe) {
583                                         found= 1;
584                                         break;
585                                 }
586                         }
587                 }
588                 else {
589                         /* find gp-frame which is less than or equal to cframe */
590                         for (gpf= gpl->frames.first; gpf; gpf= gpf->next) {
591                                 if (gpf->framenum <= cframe) {
592                                         found= 1;
593                                         break;
594                                 }
595                         }
596                 }
597                 
598                 /* set the appropriate frame */
599                 if (addnew) {
600                         if ((found) && (gpf->framenum == cframe))
601                                 gpl->actframe= gpf;
602                         else
603                                 gpl->actframe= gpencil_frame_addnew(gpl, cframe);
604                 }
605                 else if (found)
606                         gpl->actframe= gpf;
607                 else {
608                         /* unresolved errogenous situation! */
609                         printf("Error: cannot find appropriate gp-frame \n");
610                         /* gpl->actframe should still be NULL */
611                 }
612         }
613         else {
614                 /* currently no frames (add if allowed to) */
615                 if (addnew)
616                         gpl->actframe= gpencil_frame_addnew(gpl, cframe);
617                 else {
618                         /* don't do anything... this may be when no frames yet! */
619                         /* gpl->actframe should still be NULL */
620                 }
621         }
622         
623         /* return */
624         return gpl->actframe;
625 }
626
627 /* delete the given frame from a layer */
628 void gpencil_layer_delframe (bGPDlayer *gpl, bGPDframe *gpf)
629 {
630         /* error checking */
631         if (ELEM(NULL, gpl, gpf))
632                 return;
633                 
634         /* free the frame and its data */
635         free_gpencil_strokes(gpf);
636         BLI_freelinkN(&gpl->frames, gpf);
637         gpl->actframe = NULL;
638 }
639
640 /* get the active gp-layer for editing */
641 bGPDlayer *gpencil_layer_getactive (bGPdata *gpd)
642 {
643         bGPDlayer *gpl;
644         
645         /* error checking */
646         if (ELEM(NULL, gpd, gpd->layers.first))
647                 return NULL;
648                 
649         /* loop over layers until found (assume only one active) */
650         for (gpl=gpd->layers.first; gpl; gpl=gpl->next) {
651                 if (gpl->flag & GP_LAYER_ACTIVE)
652                         return gpl;
653         }
654         
655         /* no active layer found */
656         return NULL;
657 }
658
659 /* set the active gp-layer */
660 void gpencil_layer_setactive (bGPdata *gpd, bGPDlayer *active)
661 {
662         bGPDlayer *gpl;
663         
664         /* error checking */
665         if (ELEM3(NULL, gpd, gpd->layers.first, active))
666                 return;
667                 
668         /* loop over layers deactivating all */
669         for (gpl=gpd->layers.first; gpl; gpl=gpl->next)
670                 gpl->flag &= ~GP_LAYER_ACTIVE;
671         
672         /* set as active one */
673         active->flag |= GP_LAYER_ACTIVE;
674 }
675
676 /* delete the active gp-layer */
677 void gpencil_layer_delactive (bGPdata *gpd)
678 {
679         bGPDlayer *gpl= gpencil_layer_getactive(gpd);
680         
681         /* error checking */
682         if (ELEM(NULL, gpd, gpl)) 
683                 return;
684         
685         /* free layer */        
686         free_gpencil_frames(gpl);
687         BLI_freelinkN(&gpd->layers, gpl);
688 }
689
690 /* ************************************************** */
691 /* GREASE-PENCIL EDITING - Tools */
692
693 /* --------- Data Deletion ---------- */
694
695 /* delete the last stroke on the active layer */
696 void gpencil_delete_laststroke (bGPdata *gpd, int cfra)
697 {
698         bGPDlayer *gpl= gpencil_layer_getactive(gpd);
699         bGPDframe *gpf= gpencil_layer_getframe(gpl, cfra, 0);
700         
701         gpencil_frame_delete_laststroke(gpf);
702 }
703
704 /* delete the active frame */
705 void gpencil_delete_actframe (bGPdata *gpd, int cfra)
706 {
707         bGPDlayer *gpl= gpencil_layer_getactive(gpd);
708         bGPDframe *gpf= gpencil_layer_getframe(gpl, cfra, 0);
709         
710         gpencil_layer_delframe(gpl, gpf);
711 }
712
713
714
715 /* delete various grase-pencil elements 
716  *      mode:   1 - last stroke
717  *                      2 - active frame
718  *                      3 - active layer
719  */
720 void gpencil_delete_operation (int cfra, short mode)
721 {
722         bGPdata *gpd;
723         
724         /* get datablock to work on */
725         gpd= gpencil_data_getactive(NULL);
726         if (gpd == NULL) return;
727         
728         switch (mode) {
729                 case 1: /* last stroke */
730                         gpencil_delete_laststroke(gpd, cfra);
731                         break;
732                 case 2: /* active frame */
733                         gpencil_delete_actframe(gpd, cfra);
734                         break;
735                 case 3: /* active layer */
736                         gpencil_layer_delactive(gpd);
737                         break;
738         }
739         
740         /* redraw and undo-push */
741         BIF_undo_push("GPencil Delete");
742 }
743
744 /* display a menu for deleting different grease-pencil elements */
745 void gpencil_delete_menu (void)
746 {
747         bGPdata *gpd= gpencil_data_getactive(NULL);
748         int cfra= 0; // XXX
749         short mode;
750         
751         /* only show menu if it will be relevant */
752         if (gpd == NULL) return;
753         
754         mode= pupmenu("Grease Pencil Erase...%t|Last Stroke%x1|Active Frame%x2|Active Layer%x3");
755         if (mode <= 0) return;
756         
757         gpencil_delete_operation(cfra, mode);
758 }
759
760 /* --------- Data Conversion ---------- */
761
762 /* convert the coordinates from the given stroke point into 3d-coordinates */
763 static void gp_strokepoint_convertcoords (bGPDstroke *gps, bGPDspoint *pt, float p3d[3])
764 {
765         ARegion *ar= NULL; // XXX
766         
767         if (gps->flag & GP_STROKE_3DSPACE) {
768                 /* directly use 3d-coordinates */
769                 VecCopyf(p3d, &pt->x);
770         }
771         else {
772                 short mval[2];
773                 int mx, my;
774                 float *fp= give_cursor(NULL, NULL); // XXX should be scene, v3d
775                 float dvec[3];
776                 
777                 /* get screen coordinate */
778                 if (gps->flag & GP_STROKE_2DSPACE) {
779                         // XXX
780                         // View2D *v2d= spacelink_get_view2d(curarea->spacedata.first);
781                         // UI_view2d_view_to_region(v2d, pt->x, pt->y, &mx, &my);
782                 }
783                 else {
784                         // XXX
785                         // mx= (short)(pt->x / 1000 * curarea->winx);
786                         // my= (short)(pt->y / 1000 * curarea->winy);
787                 }
788                 
789                 /* convert screen coordinate to 3d coordinates 
790                  *      - method taken from editview.c - mouse_cursor() 
791                  */
792                 project_short_noclip(ar, fp, mval);
793                 window_to_3d(ar, dvec, mval[0]-mx, mval[1]-my);
794                 VecSubf(p3d, fp, dvec);
795         }
796 }
797
798 /* --- */
799
800 /* convert stroke to 3d path */
801 static void gp_stroke_to_path (bGPDlayer *gpl, bGPDstroke *gps, Curve *cu)
802 {
803         bGPDspoint *pt;
804         Nurb *nu;
805         BPoint *bp;
806         int i;
807         
808         /* create new 'nurb' within the curve */
809         nu = (Nurb *)MEM_callocN(sizeof(Nurb), "gpstroke_to_path(nurb)");
810         
811         nu->pntsu= gps->totpoints;
812         nu->pntsv= 1;
813         nu->orderu= gps->totpoints;
814         nu->flagu= 2;   /* endpoint */
815         nu->resolu= 32;
816         
817         nu->bp= (BPoint *)MEM_callocN(sizeof(BPoint)*gps->totpoints, "bpoints");
818         
819         /* add points */
820         for (i=0, pt=gps->points, bp=nu->bp; i < gps->totpoints; i++, pt++, bp++) {
821                 float p3d[3];
822                 
823                 /* get coordinates to add at */
824                 gp_strokepoint_convertcoords(gps, pt, p3d);
825                 VecCopyf(bp->vec, p3d);
826                 
827                 /* set settings */
828                 bp->f1= SELECT;
829                 bp->radius = bp->weight = pt->pressure * gpl->thickness;
830         }
831         
832         /* add nurb to curve */
833         BLI_addtail(&cu->nurb, nu);
834 }
835
836 /* convert stroke to 3d bezier */
837 static void gp_stroke_to_bezier (bGPDlayer *gpl, bGPDstroke *gps, Curve *cu)
838 {
839         bGPDspoint *pt;
840         Nurb *nu;
841         BezTriple *bezt;
842         int i;
843         
844         /* create new 'nurb' within the curve */
845         nu = (Nurb *)MEM_callocN(sizeof(Nurb), "gpstroke_to_bezier(nurb)");
846         
847         nu->pntsu= gps->totpoints;
848         nu->resolu= 12;
849         nu->resolv= 12;
850         nu->type= CU_BEZIER;
851         nu->bezt = (BezTriple *)MEM_callocN(gps->totpoints*sizeof(BezTriple), "bezts");
852         
853         /* add points */
854         for (i=0, pt=gps->points, bezt=nu->bezt; i < gps->totpoints; i++, pt++, bezt++) {
855                 float p3d[3];
856                 
857                 /* get coordinates to add at */
858                 gp_strokepoint_convertcoords(gps, pt, p3d);
859                 
860                 /* TODO: maybe in future the handles shouldn't be in same place */
861                 VecCopyf(bezt->vec[0], p3d);
862                 VecCopyf(bezt->vec[1], p3d);
863                 VecCopyf(bezt->vec[2], p3d);
864                 
865                 /* set settings */
866                 bezt->h1= bezt->h2= HD_FREE;
867                 bezt->f1= bezt->f2= bezt->f3= SELECT;
868                 bezt->radius = bezt->weight = pt->pressure * gpl->thickness * 0.1f;
869         }
870         
871         /* must calculate handles or else we crash */
872         calchandlesNurb(nu);
873         
874         /* add nurb to curve */
875         BLI_addtail(&cu->nurb, nu);
876 }
877
878 /* convert a given grease-pencil layer to a 3d-curve representation (using current view if appropriate) */
879 static void gp_layer_to_curve (bGPdata *gpd, bGPDlayer *gpl, Scene *scene, short mode)
880 {
881         bGPDframe *gpf= gpencil_layer_getframe(gpl, scene->r.cfra, 0);
882         bGPDstroke *gps;
883         Object *ob;
884         Curve *cu;
885         
886         /* error checking */
887         if (ELEM3(NULL, gpd, gpl, gpf))
888                 return;
889                 
890         /* only convert if there are any strokes on this layer's frame to convert */
891         if (gpf->strokes.first == NULL)
892                 return;
893         
894         /* init the curve object (remove rotation and get curve data from it)
895          *      - must clear transforms set on object, as those skew our results
896          */
897         add_object_draw(OB_CURVE);
898         ob= OBACT;
899         ob->loc[0]= ob->loc[1]= ob->loc[2]= 0;
900         ob->rot[0]= ob->rot[1]= ob->rot[2]= 0;
901         cu= ob->data;
902         cu->flag |= CU_3D;
903         
904         /* rename object and curve to layer name */
905         rename_id((ID *)ob, gpl->info);
906         rename_id((ID *)cu, gpl->info);
907         
908         /* add points to curve */
909         for (gps= gpf->strokes.first; gps; gps= gps->next) {
910                 switch (mode) {
911                         case 1: 
912                                 gp_stroke_to_path(gpl, gps, cu);
913                                 break;
914                         case 2:
915                                 gp_stroke_to_bezier(gpl, gps, cu);
916                                 break;
917                 }
918         }
919 }
920
921 /* --- */
922
923 /* convert a stroke to a bone chain */
924 static void gp_stroke_to_bonechain (bGPDlayer *gpl, bGPDstroke *gps, bArmature *arm, ListBase *bones)
925 {
926         EditBone *ebo, *prev=NULL;
927         bGPDspoint *pt, *ptn;
928         int i;
929         
930         /* add each segment separately */
931         for (i=0, pt=gps->points, ptn=gps->points+1; i < (gps->totpoints-1); prev=ebo, i++, pt++, ptn++) {
932                 float p3da[3], p3db[3];
933                 
934                 /* get coordinates to add at */
935                 gp_strokepoint_convertcoords(gps, pt, p3da);
936                 gp_strokepoint_convertcoords(gps, ptn, p3db);
937                 
938                 /* allocate new bone */
939                 ebo= MEM_callocN(sizeof(EditBone), "eBone");
940                 
941                 VecCopyf(ebo->head, p3da);
942                 VecCopyf(ebo->tail, p3db);
943                 
944                 /* add new bone - note: sync with editarmature.c::add_editbone() */
945                 {
946                         BLI_strncpy(ebo->name, "Stroke", 32);
947                         unique_editbone_name(bones, ebo->name);
948                         
949                         BLI_addtail(bones, ebo);
950                         
951                         if (i > 0)
952                         {
953                                 ebo->flag |= BONE_CONNECTED;
954                         }
955                         ebo->weight= 1.0f;
956                         ebo->dist= 0.25f;
957                         ebo->xwidth= 0.1f;
958                         ebo->zwidth= 0.1f;
959                         ebo->ease1= 1.0f;
960                         ebo->ease2= 1.0f;
961                         ebo->rad_head= pt->pressure * gpl->thickness * 0.1f;
962                         ebo->rad_tail= ptn->pressure * gpl->thickness * 0.1f;
963                         ebo->segments= 1;
964                         ebo->layer= arm->layer;
965                 }
966                 
967                 /* set parenting */
968                 ebo->parent= prev;
969         }
970 }
971
972 /* convert a given grease-pencil layer to a 3d-curve representation (using current view if appropriate) */
973 static void gp_layer_to_armature (bGPdata *gpd, bGPDlayer *gpl, Scene *scene, View3D *v3d, short mode)
974 {
975         bGPDframe *gpf= gpencil_layer_getframe(gpl, scene->r.cfra, 0);
976         bGPDstroke *gps;
977         Object *ob;
978         bArmature *arm;
979         
980         /* error checking */
981         if (ELEM3(NULL, gpd, gpl, gpf))
982                 return;
983                 
984         /* only convert if there are any strokes on this layer's frame to convert */
985         if (gpf->strokes.first == NULL)
986                 return;
987         
988         /* init the armature object (remove rotation and assign armature data to it) 
989          *      - must clear transforms set on object, as those skew our results
990          */
991         add_object_draw(OB_ARMATURE);
992         ob= OBACT;
993         ob->loc[0]= ob->loc[1]= ob->loc[2]= 0;
994         ob->rot[0]= ob->rot[1]= ob->rot[2]= 0;
995         arm= ob->data;
996         
997         /* rename object and armature to layer name */
998         rename_id((ID *)ob, gpl->info);
999         rename_id((ID *)arm, gpl->info);
1000         
1001         /* this is editmode armature */
1002         arm->edbo= MEM_callocN(sizeof(ListBase), "arm edbo");
1003         
1004         /* convert segments to bones, strokes to bone chains */
1005         for (gps= gpf->strokes.first; gps; gps= gps->next) {
1006                 gp_stroke_to_bonechain(gpl, gps, arm, arm->edbo);
1007         }
1008         
1009         /* adjust roll of bones
1010          *      - set object as EditMode object, but need to clear afterwards!
1011          *      - use 'align to world z-up' option
1012          */
1013         {
1014                 /* set our data as if we're in editmode to fool auto_align_armature() */
1015                 scene->obedit= ob;
1016                 
1017                 /* WARNING: need to make sure this magic number doesn't change */
1018                 auto_align_armature(scene, v3d, 2);     
1019                 
1020                 scene->obedit= NULL;
1021         }
1022         
1023         /* flush editbones to armature */
1024         ED_armature_from_edit(scene, ob);
1025         ED_armature_edit_free(ob);
1026 }
1027
1028 /* --- */
1029
1030 /* convert grease-pencil strokes to another representation 
1031  *      mode:   1 - Active layer to path
1032  *                      2 - Active layer to bezier
1033  *                      3 - Active layer to armature
1034  */
1035 void gpencil_convert_operation (short mode)
1036 {
1037         Scene *scene= NULL; // XXX
1038         View3D *v3d= NULL; // XXX
1039         RegionView3D *rv3d= NULL; // XXX
1040         bGPdata *gpd;   
1041         float *fp= give_cursor(scene, v3d);
1042         
1043         /* get datablock to work on */
1044         gpd= gpencil_data_getactive(NULL);
1045         if (gpd == NULL) return;
1046         
1047         /* initialise 3d-cursor correction globals */
1048         initgrabz(rv3d, fp[0], fp[1], fp[2]);
1049         
1050         /* handle selection modes */
1051         switch (mode) {
1052                 case 1: /* active layer only (to path) */
1053                 case 2: /* active layer only (to bezier) */
1054                 {
1055                         bGPDlayer *gpl= gpencil_layer_getactive(gpd);
1056                         gp_layer_to_curve(gpd, gpl, scene, mode);
1057                 }
1058                         break;
1059                 case 3: /* active layer only (to armature) */
1060                 {
1061                         bGPDlayer *gpl= gpencil_layer_getactive(gpd);
1062                         gp_layer_to_armature(gpd, gpl, scene, v3d, mode);
1063                 }
1064                         break;
1065         }
1066         
1067         /* redraw and undo-push */
1068         BIF_undo_push("GPencil Convert");
1069 }
1070
1071 /* display a menu for converting grease-pencil strokes */
1072 void gpencil_convert_menu (void)
1073 {
1074         bGPdata *gpd= gpencil_data_getactive(NULL);
1075         short mode;
1076         
1077         /* only show menu if it will be relevant */
1078         if (gpd == NULL) return;
1079         
1080         mode= pupmenu("Grease Pencil Convert %t|Active Layer To Path%x1|Active Layer to Bezier%x2|Active Layer to Armature%x3");
1081         if (mode <= 0) return;
1082         
1083         gpencil_convert_operation(mode);
1084 }
1085
1086 /* ************************************************** */
1087 /* GREASE-PENCIL EDITING MODE - Painting */
1088
1089 /* ---------- 'Globals' and Defines ----------------- */
1090
1091 /* maximum sizes of gp-session buffer */
1092 #define GP_STROKE_BUFFER_MAX    5000
1093
1094 /* Macros for accessing sensitivity thresholds... */
1095         /* minimum number of pixels mouse should move before new point created */
1096 #define MIN_MANHATTEN_PX        (U.gp_manhattendist)
1097         /* minimum length of new segment before new point can be added */
1098 #define MIN_EUCLIDEAN_PX        (U.gp_euclideandist)
1099
1100 /* macro to test if only converting endpoints - only for use when converting!  */       
1101 #define GP_BUFFER2STROKE_ENDPOINTS ((gpd->flag & GP_DATA_EDITPAINT) && (ctrl))
1102         
1103 /* ------ */
1104
1105 /* Temporary 'Stroke' Operation data */
1106 typedef struct tGPsdata {
1107         Scene *scene;       /* current scene from context */
1108         ScrArea *sa;            /* area where painting originated */
1109         ARegion *ar;        /* region where painting originated */
1110         View2D *v2d;            /* needed for GP_STROKE_2DSPACE */
1111         
1112         ImBuf *ibuf;            /* needed for GP_STROKE_2DIMAGE */
1113         struct IBufViewSettings {
1114                 int offsx, offsy;                       /* offsets */
1115                 int sizex, sizey;                       /* dimensions to use as scale-factor */
1116         } im2d_settings;        /* needed for GP_STROKE_2DIMAGE */
1117         
1118         bGPdata *gpd;           /* gp-datablock layer comes from */
1119         bGPDlayer *gpl;         /* layer we're working on */
1120         bGPDframe *gpf;         /* frame we're working on */
1121         
1122         short status;           /* current status of painting */
1123         short paintmode;        /* mode for painting */
1124         
1125         short mval[2];          /* current mouse-position */
1126         short mvalo[2];         /* previous recorded mouse-position */
1127         
1128         float pressure;         /* current stylus pressure */
1129         float opressure;        /* previous stylus pressure */
1130         
1131         short radius;           /* radius of influence for eraser */
1132 } tGPsdata;
1133
1134 /* values for tGPsdata->status */
1135 enum {
1136         GP_STATUS_NORMAL = 0,   /* running normally */
1137         GP_STATUS_ERROR,                /* something wasn't correctly set up */
1138         GP_STATUS_DONE                  /* painting done */
1139 };
1140
1141 /* values for tGPsdata->paintmode */
1142 enum {
1143         GP_PAINTMODE_DRAW = 0,
1144         GP_PAINTMODE_ERASER
1145 };
1146
1147 /* Return flags for adding points to stroke buffer */
1148 enum {
1149         GP_STROKEADD_INVALID    = -2,           /* error occurred - insufficient info to do so */
1150         GP_STROKEADD_OVERFLOW   = -1,           /* error occurred - cannot fit any more points */
1151         GP_STROKEADD_NORMAL,                            /* point was successfully added */
1152         GP_STROKEADD_FULL                                       /* cannot add any more points to buffer */
1153 };
1154
1155 /* ---------- Stroke Editing ------------ */
1156
1157 /* clear the session buffers (call this before AND after a paint operation) */
1158 static void gp_session_validatebuffer (tGPsdata *p)
1159 {
1160         bGPdata *gpd= p->gpd;
1161         
1162         /* clear memory of buffer (or allocate it if starting a new session) */
1163         if (gpd->sbuffer)
1164                 memset(gpd->sbuffer, 0, sizeof(tGPspoint)*GP_STROKE_BUFFER_MAX);
1165         else
1166                 gpd->sbuffer= MEM_callocN(sizeof(tGPspoint)*GP_STROKE_BUFFER_MAX, "gp_session_strokebuffer");
1167         
1168         /* reset indices */
1169         gpd->sbuffer_size = 0;
1170         
1171         /* reset flags */
1172         gpd->sbuffer_sflag= 0;
1173 }
1174
1175 /* check if the current mouse position is suitable for adding a new point */
1176 static short gp_stroke_filtermval (tGPsdata *p, short mval[2], short pmval[2])
1177 {
1178         short dx= abs(mval[0] - pmval[0]);
1179         short dy= abs(mval[1] - pmval[1]);
1180         
1181         /* check if mouse moved at least certain distance on both axes (best case) */
1182         if ((dx > MIN_MANHATTEN_PX) && (dy > MIN_MANHATTEN_PX))
1183                 return 1;
1184         
1185         /* check if the distance since the last point is significant enough */
1186         // future optimisation: sqrt here may be too slow?
1187         else if (sqrt(dx*dx + dy*dy) > MIN_EUCLIDEAN_PX)
1188                 return 1;
1189         
1190         /* mouse 'didn't move' */
1191         else
1192                 return 0;
1193 }
1194
1195 /* convert screen-coordinates to buffer-coordinates */
1196 static void gp_stroke_convertcoords (tGPsdata *p, short mval[], float out[])
1197 {
1198         bGPdata *gpd= p->gpd;
1199         
1200         /* in 3d-space - pt->x/y/z are 3 side-by-side floats */
1201         if (gpd->sbuffer_sflag & GP_STROKE_3DSPACE) {
1202                 const short mx=mval[0], my=mval[1];
1203                 float *fp= give_cursor(p->scene, NULL); // XXX NULL could be v3d
1204                 float dvec[3];
1205                 
1206                 /* Current method just converts each point in screen-coordinates to 
1207                  * 3D-coordinates using the 3D-cursor as reference. In general, this 
1208                  * works OK, but it could of course be improved.
1209                  *
1210                  * TODO:
1211                  *      - investigate using nearest point(s) on a previous stroke as
1212                  *        reference point instead or as offset, for easier stroke matching
1213                  *      - investigate projection onto geometry (ala retopo)
1214                  */
1215                 
1216                 /* method taken from editview.c - mouse_cursor() */
1217                 project_short_noclip(p->ar, fp, mval);
1218                 window_to_3d(p->ar, dvec, mval[0]-mx, mval[1]-my);
1219                 VecSubf(out, fp, dvec);
1220         }
1221         
1222         /* 2d - on 'canvas' (assume that p->v2d is set) */
1223         else if ((gpd->sbuffer_sflag & GP_STROKE_2DSPACE) && (p->v2d)) {
1224                 float x, y;
1225                 
1226                 UI_view2d_region_to_view(p->v2d, mval[0], mval[1], &x, &y);
1227                 
1228                 out[0]= x;
1229                 out[1]= y;
1230         }
1231         
1232         /* 2d - on image 'canvas' (assume that p->v2d is set) */
1233         else if (gpd->sbuffer_sflag & GP_STROKE_2DIMAGE) {
1234                 int sizex, sizey, offsx, offsy;
1235                 
1236                 /* get stored settings 
1237                  *      - assume that these have been set already (there are checks that set sane 'defaults' just in case)
1238                  */
1239                 sizex= p->im2d_settings.sizex;
1240                 sizey= p->im2d_settings.sizey;
1241                 offsx= p->im2d_settings.offsx;
1242                 offsy= p->im2d_settings.offsy;
1243                 
1244                 /* calculate new points */
1245                 out[0]= (float)(mval[0] - offsx) / (float)sizex;
1246                 out[1]= (float)(mval[1] - offsy) / (float)sizey;
1247         }
1248         
1249         /* 2d - relative to screen (viewport area) */
1250         else {
1251                 out[0] = (float)(mval[0]) / (float)(p->sa->winx) * 1000;
1252                 out[1] = (float)(mval[1]) / (float)(p->sa->winy) * 1000;
1253         }
1254 }
1255
1256 /* add current stroke-point to buffer (returns whether point was successfully added) */
1257 static short gp_stroke_addpoint (tGPsdata *p, short mval[2], float pressure)
1258 {
1259         bGPdata *gpd= p->gpd;
1260         tGPspoint *pt;
1261         
1262         /* check if still room in buffer */
1263         if (gpd->sbuffer_size >= GP_STROKE_BUFFER_MAX)
1264                 return GP_STROKEADD_OVERFLOW;
1265         
1266         /* get pointer to destination point */
1267         pt= ((tGPspoint *)(gpd->sbuffer) + gpd->sbuffer_size);
1268         
1269         /* store settings */
1270         pt->x= mval[0];
1271         pt->y= mval[1];
1272         pt->pressure= pressure;
1273         
1274         /* increment counters */
1275         gpd->sbuffer_size++;
1276         
1277         /* check if another operation can still occur */
1278         if (gpd->sbuffer_size == GP_STROKE_BUFFER_MAX)
1279                 return GP_STROKEADD_FULL;
1280         else
1281                 return GP_STROKEADD_NORMAL;
1282 }
1283
1284 /* smooth a stroke (in buffer) before storing it */
1285 static void gp_stroke_smooth (tGPsdata *p)
1286 {
1287         bGPdata *gpd= p->gpd;
1288         int i=0, cmx=gpd->sbuffer_size;
1289         int ctrl= 0; // XXX
1290         
1291         /* only smooth if smoothing is enabled, and we're not doing a straight line */
1292         if (!(U.gp_settings & GP_PAINT_DOSMOOTH) || GP_BUFFER2STROKE_ENDPOINTS)
1293                 return;
1294         
1295         /* don't try if less than 2 points in buffer */
1296         if ((cmx <= 2) || (gpd->sbuffer == NULL))
1297                 return;
1298         
1299         /* apply weighting-average (note doing this along path sequentially does introduce slight error) */
1300         for (i=0; i < gpd->sbuffer_size; i++) {
1301                 tGPspoint *pc= (((tGPspoint *)gpd->sbuffer) + i);
1302                 tGPspoint *pb= (i-1 > 0)?(pc-1):(pc);
1303                 tGPspoint *pa= (i-2 > 0)?(pc-2):(pb);
1304                 tGPspoint *pd= (i+1 < cmx)?(pc+1):(pc);
1305                 tGPspoint *pe= (i+2 < cmx)?(pc+2):(pd);
1306                 
1307                 pc->x= (short)(0.1*pa->x + 0.2*pb->x + 0.4*pc->x + 0.2*pd->x + 0.1*pe->x);
1308                 pc->y= (short)(0.1*pa->y + 0.2*pb->y + 0.4*pc->y + 0.2*pd->y + 0.1*pe->y);
1309         }
1310 }
1311
1312 /* simplify a stroke (in buffer) before storing it 
1313  *      - applies a reverse Chaikin filter
1314  *      - code adapted from etch-a-ton branch (editarmature_sketch.c)
1315  */
1316 static void gp_stroke_simplify (tGPsdata *p)
1317 {
1318         bGPdata *gpd= p->gpd;
1319         tGPspoint *old_points= (tGPspoint *)gpd->sbuffer;
1320         short num_points= gpd->sbuffer_size;
1321         short flag= gpd->sbuffer_sflag;
1322         short i, j;
1323         int ctrl= 0; // XXX
1324         
1325         /* only simplify if simlification is enabled, and we're not doing a straight line */
1326         if (!(U.gp_settings & GP_PAINT_DOSIMPLIFY) || GP_BUFFER2STROKE_ENDPOINTS)
1327                 return;
1328         
1329         /* don't simplify if less than 4 points in buffer */
1330         if ((num_points <= 2) || (old_points == NULL))
1331                 return;
1332                 
1333         /* clear buffer (but don't free mem yet) so that we can write to it 
1334          *      - firstly set sbuffer to NULL, so a new one is allocated
1335          *      - secondly, reset flag after, as it gets cleared auto
1336          */
1337         gpd->sbuffer= NULL;
1338         gp_session_validatebuffer(p);
1339         gpd->sbuffer_sflag = flag;
1340         
1341 /* macro used in loop to get position of new point
1342  *      - used due to the mixture of datatypes in use here
1343  */
1344 #define GP_SIMPLIFY_AVPOINT(offs, sfac) \
1345         { \
1346                 co[0] += (float)(old_points[offs].x * sfac); \
1347                 co[1] += (float)(old_points[offs].y * sfac); \
1348                 pressure += old_points[offs].pressure * sfac; \
1349         }
1350         
1351         for (i = 0, j = 0; i < num_points; i++)
1352         {
1353                 if (i - j == 3)
1354                 {
1355                         float co[2], pressure;
1356                         short mco[2];
1357                         
1358                         /* initialise values */
1359                         co[0]= 0;
1360                         co[1]= 0;
1361                         pressure = 0;
1362                         
1363                         /* using macro, calculate new point */
1364                         GP_SIMPLIFY_AVPOINT(j, -0.25f);
1365                         GP_SIMPLIFY_AVPOINT(j+1, 0.75f);
1366                         GP_SIMPLIFY_AVPOINT(j+2, 0.75f);
1367                         GP_SIMPLIFY_AVPOINT(j+3, -0.25f);
1368                         
1369                         /* set values for adding */
1370                         mco[0]= (short)co[0];
1371                         mco[1]= (short)co[1];
1372                         
1373                         /* ignore return values on this... assume to be ok for now */
1374                         gp_stroke_addpoint(p, mco, pressure);
1375                         
1376                         j += 2;
1377                 }
1378         } 
1379         
1380         /* free old buffer */
1381         MEM_freeN(old_points);
1382 }
1383
1384
1385 /* make a new stroke from the buffer data */
1386 static void gp_stroke_newfrombuffer (tGPsdata *p)
1387 {
1388         bGPdata *gpd= p->gpd;
1389         bGPDstroke *gps;
1390         bGPDspoint *pt;
1391         tGPspoint *ptc;
1392         int i, totelem;
1393         int ctrl= 0; // XXX
1394         
1395         /* get total number of points to allocate space for:
1396          *      - in 'Draw Mode', holding the Ctrl-Modifier will only take endpoints
1397          *      - otherwise, do whole stroke
1398          */
1399         if (GP_BUFFER2STROKE_ENDPOINTS)
1400                 totelem = (gpd->sbuffer_size >= 2) ? 2: gpd->sbuffer_size;
1401         else
1402                 totelem = gpd->sbuffer_size;
1403         
1404         /* exit with error if no valid points from this stroke */
1405         if (totelem == 0) {
1406                 if (G.f & G_DEBUG) 
1407                         printf("Error: No valid points in stroke buffer to convert (tot=%d) \n", gpd->sbuffer_size);
1408                 return;
1409         }
1410         
1411         /* allocate memory for a new stroke */
1412         gps= MEM_callocN(sizeof(bGPDstroke), "gp_stroke");
1413         
1414         /* allocate enough memory for a continuous array for storage points */
1415         pt= gps->points= MEM_callocN(sizeof(bGPDspoint)*totelem, "gp_stroke_points");
1416         
1417         /* copy appropriate settings for stroke */
1418         gps->totpoints= totelem;
1419         gps->thickness= p->gpl->thickness;
1420         gps->flag= gpd->sbuffer_sflag;
1421         
1422         /* copy points from the buffer to the stroke */
1423         if (GP_BUFFER2STROKE_ENDPOINTS) {
1424                 /* 'Draw Mode' + Ctrl-Modifier - only endpoints */
1425                 {
1426                         /* first point */
1427                         ptc= gpd->sbuffer;
1428                         
1429                         /* convert screen-coordinates to appropriate coordinates (and store them) */
1430                         gp_stroke_convertcoords(p, &ptc->x, &pt->x);
1431                         
1432                         /* copy pressure */
1433                         pt->pressure= ptc->pressure;
1434                         
1435                         pt++;
1436                 }
1437                         
1438                 if (totelem == 2) {
1439                         /* last point if applicable */
1440                         ptc= ((tGPspoint *)gpd->sbuffer) + (gpd->sbuffer_size - 1);
1441                         
1442                         /* convert screen-coordinates to appropriate coordinates (and store them) */
1443                         gp_stroke_convertcoords(p, &ptc->x, &pt->x);
1444                         
1445                         /* copy pressure */
1446                         pt->pressure= ptc->pressure;
1447                 }
1448         }
1449         else {
1450                 /* convert all points (normal behaviour) */
1451                 for (i=0, ptc=gpd->sbuffer; i < gpd->sbuffer_size && ptc; i++, ptc++) {
1452                         /* convert screen-coordinates to appropriate coordinates (and store them) */
1453                         gp_stroke_convertcoords(p, &ptc->x, &pt->x);
1454                         
1455                         /* copy pressure */
1456                         pt->pressure= ptc->pressure;
1457                         
1458                         pt++;
1459                 }
1460         }
1461         
1462         /* add stroke to frame */
1463         BLI_addtail(&p->gpf->strokes, gps);
1464 }
1465
1466 /* --- 'Eraser' for 'Paint' Tool ------ */
1467
1468 /* eraser tool - remove segment from stroke/split stroke (after lasso inside) */
1469 static short gp_stroke_eraser_splitdel (bGPDframe *gpf, bGPDstroke *gps, int i)
1470 {
1471         bGPDspoint *pt_tmp= gps->points;
1472         bGPDstroke *gsn = NULL;
1473
1474         /* if stroke only had two points, get rid of stroke */
1475         if (gps->totpoints == 2) {
1476                 /* free stroke points, then stroke */
1477                 MEM_freeN(pt_tmp);
1478                 BLI_freelinkN(&gpf->strokes, gps);
1479                 
1480                 /* nothing left in stroke, so stop */
1481                 return 1;
1482         }
1483
1484         /* if last segment, just remove segment from the stroke */
1485         else if (i == gps->totpoints - 2) {
1486                 /* allocate new points array, and assign most of the old stroke there */
1487                 gps->totpoints--;
1488                 gps->points= MEM_callocN(sizeof(bGPDspoint)*gps->totpoints, "gp_stroke_points");
1489                 memcpy(gps->points, pt_tmp, sizeof(bGPDspoint)*gps->totpoints);
1490                 
1491                 /* free temp buffer */
1492                 MEM_freeN(pt_tmp);
1493                 
1494                 /* nothing left in stroke, so stop */
1495                 return 1;
1496         }
1497
1498         /* if first segment, just remove segment from the stroke */
1499         else if (i == 0) {
1500                 /* allocate new points array, and assign most of the old stroke there */
1501                 gps->totpoints--;
1502                 gps->points= MEM_callocN(sizeof(bGPDspoint)*gps->totpoints, "gp_stroke_points");
1503                 memcpy(gps->points, pt_tmp + 1, sizeof(bGPDspoint)*gps->totpoints);
1504                 
1505                 /* free temp buffer */
1506                 MEM_freeN(pt_tmp);
1507                 
1508                 /* no break here, as there might still be stuff to remove in this stroke */
1509                 return 0;
1510         }
1511
1512         /* segment occurs in 'middle' of stroke, so split */
1513         else {
1514                 /* duplicate stroke, and assign 'later' data to that stroke */
1515                 gsn= MEM_dupallocN(gps);
1516                 gsn->prev= gsn->next= NULL;
1517                 BLI_insertlinkafter(&gpf->strokes, gps, gsn);
1518                 
1519                 gsn->totpoints= gps->totpoints - i;
1520                 gsn->points= MEM_callocN(sizeof(bGPDspoint)*gsn->totpoints, "gp_stroke_points");
1521                 memcpy(gsn->points, pt_tmp + i, sizeof(bGPDspoint)*gsn->totpoints);
1522                 
1523                 /* adjust existing stroke  */
1524                 gps->totpoints= i;
1525                 gps->points= MEM_callocN(sizeof(bGPDspoint)*gps->totpoints, "gp_stroke_points");
1526                 memcpy(gps->points, pt_tmp, sizeof(bGPDspoint)*i);
1527                 
1528                 /* free temp buffer */
1529                 MEM_freeN(pt_tmp);
1530                 
1531                 /* nothing left in stroke, so stop */
1532                 return 1;
1533         }
1534 }
1535
1536 /* eraser tool - check if part of stroke occurs within last segment drawn by eraser */
1537 static short gp_stroke_eraser_strokeinside (short mval[], short mvalo[], short rad, short x0, short y0, short x1, short y1)
1538 {
1539         /* simple within-radius check for now */
1540         if (edge_inside_circle(mval[0], mval[1], rad, x0, y0, x1, y1))
1541                 return 1;
1542         
1543         /* not inside */
1544         return 0;
1545
1546
1547 /* eraser tool - evaluation per stroke */
1548 static void gp_stroke_eraser_dostroke (tGPsdata *p, short mval[], short mvalo[], short rad, rcti *rect, bGPDframe *gpf, bGPDstroke *gps)
1549 {
1550         bGPDspoint *pt1, *pt2;
1551         int x0=0, y0=0, x1=0, y1=0;
1552         short xyval[2];
1553         int i;
1554         
1555         if (gps->totpoints == 0) {
1556                 /* just free stroke */
1557                 if (gps->points) 
1558                         MEM_freeN(gps->points);
1559                 BLI_freelinkN(&gpf->strokes, gps);
1560         }
1561         else if (gps->totpoints == 1) {
1562                 /* get coordinates */
1563                 if (gps->flag & GP_STROKE_3DSPACE) {
1564                         project_short(p->ar, &gps->points->x, xyval);
1565                         x0= xyval[0];
1566                         y0= xyval[1];
1567                 }
1568                 else if (gps->flag & GP_STROKE_2DSPACE) {                       
1569                         UI_view2d_view_to_region(p->v2d, gps->points->x, gps->points->y, &x0, &y0);
1570                 }
1571                 else if (gps->flag & GP_STROKE_2DIMAGE) {                       
1572                         int offsx, offsy, sizex, sizey;
1573                         
1574                         /* get stored settings */
1575                         sizex= p->im2d_settings.sizex;
1576                         sizey= p->im2d_settings.sizey;
1577                         offsx= p->im2d_settings.offsx;
1578                         offsy= p->im2d_settings.offsy;
1579                         
1580                         /* calculate new points */
1581                         x0= (short)((gps->points->x * sizex) + offsx);
1582                         y0= (short)((gps->points->y * sizey) + offsy);
1583                 }
1584                 else {
1585                         x0= (short)(gps->points->x / 1000 * p->sa->winx);
1586                         y0= (short)(gps->points->y / 1000 * p->sa->winy);
1587                 }
1588                 
1589                 /* do boundbox check first */
1590                 if (BLI_in_rcti(rect, x0, y0)) {
1591                         /* only check if point is inside */
1592                         if ( ((x0-mval[0])*(x0-mval[0]) + (y0-mval[1])*(y0-mval[1])) <= rad*rad ) {
1593                                 /* free stroke */
1594                                 MEM_freeN(gps->points);
1595                                 BLI_freelinkN(&gpf->strokes, gps);
1596                         }
1597                 }
1598         }
1599         else {  
1600                 /* loop over the points in the stroke, checking for intersections 
1601                  *      - an intersection will require the stroke to be split
1602                  */
1603                 for (i=0; (i+1) < gps->totpoints; i++) {
1604                         /* get points to work with */
1605                         pt1= gps->points + i;
1606                         pt2= gps->points + i + 1;
1607                         
1608                         /* get coordinates */
1609                         if (gps->flag & GP_STROKE_3DSPACE) {
1610                                 project_short(p->ar, &pt1->x, xyval);
1611                                 x0= xyval[0];
1612                                 y0= xyval[1];
1613                                 
1614                                 project_short(p->ar, &pt2->x, xyval);
1615                                 x1= xyval[0];
1616                                 y1= xyval[1];
1617                         }
1618                         else if (gps->flag & GP_STROKE_2DSPACE) {
1619                                 UI_view2d_view_to_region(p->v2d, pt1->x, pt1->y, &x0, &y0);
1620                                 
1621                                 UI_view2d_view_to_region(p->v2d, pt2->x, pt2->y, &x1, &y1);
1622                         }
1623                         else if (gps->flag & GP_STROKE_2DIMAGE) {
1624                                 int offsx, offsy, sizex, sizey;
1625                                 
1626                                 /* get stored settings */
1627                                 sizex= p->im2d_settings.sizex;
1628                                 sizey= p->im2d_settings.sizey;
1629                                 offsx= p->im2d_settings.offsx;
1630                                 offsy= p->im2d_settings.offsy;
1631                                 
1632                                 /* calculate new points */
1633                                 x0= (short)((pt1->x * sizex) + offsx);
1634                                 y0= (short)((pt1->y * sizey) + offsy);
1635                                 
1636                                 x1= (short)((pt2->x * sizex) + offsx);
1637                                 y1= (short)((pt2->y * sizey) + offsy);
1638                         }
1639                         else {
1640                                 x0= (short)(pt1->x / 1000 * p->sa->winx);
1641                                 y0= (short)(pt1->y / 1000 * p->sa->winy);
1642                                 x1= (short)(pt2->x / 1000 * p->sa->winx);
1643                                 y1= (short)(pt2->y / 1000 * p->sa->winy);
1644                         }
1645                         
1646                         /* check that point segment of the boundbox of the eraser stroke */
1647                         if (BLI_in_rcti(rect, x0, y0) || BLI_in_rcti(rect, x1, y1)) {
1648                                 /* check if point segment of stroke had anything to do with
1649                                  * eraser region  (either within stroke painted, or on its lines)
1650                                  *      - this assumes that linewidth is irrelevant
1651                                  */
1652                                 if (gp_stroke_eraser_strokeinside(mval, mvalo, rad, x0, y0, x1, y1)) {
1653                                         /* if function returns true, break this loop (as no more point to check) */
1654                                         if (gp_stroke_eraser_splitdel(gpf, gps, i))
1655                                                 break;
1656                                 }
1657                         }
1658                 }
1659         }
1660 }
1661
1662 /* erase strokes which fall under the eraser strokes */
1663 static void gp_stroke_doeraser (tGPsdata *p)
1664 {
1665         bGPDframe *gpf= p->gpf;
1666         bGPDstroke *gps, *gpn;
1667         rcti rect;
1668         
1669         /* rect is rectangle of eraser */
1670         rect.xmin= p->mval[0] - p->radius;
1671         rect.ymin= p->mval[1] - p->radius;
1672         rect.xmax= p->mval[0] + p->radius;
1673         rect.ymax= p->mval[1] + p->radius;
1674         
1675         /* loop over strokes, checking segments for intersections */
1676         for (gps= gpf->strokes.first; gps; gps= gpn) {
1677                 gpn= gps->next;
1678                 gp_stroke_eraser_dostroke(p, p->mval, p->mvalo, p->radius, &rect, gpf, gps);
1679         }
1680 }
1681
1682 /* ---------- 'Paint' Tool ------------ */
1683
1684 /* init new painting session */
1685 static void gp_session_initpaint (bContext *C, tGPsdata *p)
1686 {
1687         ScrArea *curarea= CTX_wm_area(C);
1688         ARegion *ar= CTX_wm_region(C);
1689         
1690         /* clear previous data (note: is on stack) */
1691         memset(p, 0, sizeof(tGPsdata));
1692         
1693         /* make sure the active view (at the starting time) is a 3d-view */
1694         if (curarea == NULL) {
1695                 p->status= GP_STATUS_ERROR;
1696                 if (G.f & G_DEBUG) 
1697                         printf("Error: No active view for painting \n");
1698                 return;
1699         }
1700         
1701         /* pass on current scene */
1702         p->scene= CTX_data_scene(C);
1703         
1704         switch (curarea->spacetype) {
1705                 /* supported views first */
1706                 case SPACE_VIEW3D:
1707                 {
1708                         View3D *v3d= curarea->spacedata.first;
1709                         
1710                         /* set current area */
1711                         p->sa= curarea;
1712                         p->ar= ar;
1713                         
1714                         /* check that gpencil data is allowed to be drawn */
1715                         if ((v3d->flag2 & V3D_DISPGP)==0) {
1716                                 p->status= GP_STATUS_ERROR;
1717                                 if (G.f & G_DEBUG) 
1718                                         printf("Error: In active view, Grease Pencil not shown \n");
1719                                 return;
1720                         }
1721                 }
1722                         break;
1723                 case SPACE_NODE:
1724                 {
1725                         SpaceNode *snode= curarea->spacedata.first;
1726                         
1727                         /* set current area */
1728                         p->sa= curarea;
1729                         p->ar= ar;
1730                         p->v2d= &ar->v2d;
1731                         
1732                         /* check that gpencil data is allowed to be drawn */
1733                         if ((snode->flag & SNODE_DISPGP)==0) {
1734                                 p->status= GP_STATUS_ERROR;
1735                                 if (G.f & G_DEBUG) 
1736                                         printf("Error: In active view, Grease Pencil not shown \n");
1737                                 return;
1738                         }
1739                 }
1740                         break;
1741                 case SPACE_SEQ:
1742                 {
1743                         SpaceSeq *sseq= curarea->spacedata.first;
1744                         
1745                         /* set current area */
1746                         p->sa= curarea;
1747                         p->ar= ar;
1748                         p->v2d= &ar->v2d;
1749                         
1750                         /* check that gpencil data is allowed to be drawn */
1751                         if (sseq->mainb == 0) {
1752                                 p->status= GP_STATUS_ERROR;
1753                                 if (G.f & G_DEBUG) 
1754                                         printf("Error: In active view (sequencer), active mode doesn't support Grease Pencil \n");
1755                                 return;
1756                         }
1757                         if ((sseq->flag & SEQ_DRAW_GPENCIL)==0) {
1758                                 p->status= GP_STATUS_ERROR;
1759                                 if (G.f & G_DEBUG) 
1760                                         printf("Error: In active view, Grease Pencil not shown \n");
1761                                 return;
1762                         }
1763                 }
1764                         break;  
1765                 case SPACE_IMAGE:
1766                 {
1767                         SpaceImage *sima= curarea->spacedata.first;
1768                         
1769                         /* set the current area */
1770                         p->sa= curarea;
1771                         p->ar= ar;
1772                         p->v2d= &ar->v2d;
1773                         p->ibuf= BKE_image_get_ibuf(sima->image, &sima->iuser);
1774                         
1775                         /* check that gpencil data is allowed to be drawn */
1776                         if ((sima->flag & SI_DISPGP)==0) {
1777                                 p->status= GP_STATUS_ERROR;
1778                                 if (G.f & G_DEBUG)
1779                                         printf("Error: In active view, Grease Pencil not shown \n");
1780                                 return;
1781                         }
1782                 }
1783                         break;
1784                 /* unsupported views */
1785                 default:
1786                 {
1787                         p->status= GP_STATUS_ERROR;
1788                         if (G.f & G_DEBUG) 
1789                                 printf("Error: Active view not appropriate for Grease Pencil drawing \n");
1790                         return;
1791                 }
1792                         break;
1793         }
1794         
1795         /* get gp-data */
1796         p->gpd= gpencil_data_getactive(p->sa);
1797         if (p->gpd == NULL) {
1798                 short ok;
1799                 
1800                 p->gpd= gpencil_data_addnew();
1801                 ok= gpencil_data_setactive(p->sa, p->gpd);
1802                 
1803                 /* most of the time, the following check isn't needed */
1804                 if (ok == 0) {
1805                         /* free gpencil data as it can't be used */
1806                         free_gpencil_data(p->gpd);
1807                         p->gpd= NULL;
1808                         p->status= GP_STATUS_ERROR;
1809                         if (G.f & G_DEBUG) 
1810                                 printf("Error: Could not assign newly created Grease Pencil data to active area \n");
1811                         return;
1812                 }
1813         }
1814         
1815         /* set edit flags */
1816         G.f |= G_GREASEPENCIL;
1817         
1818         /* clear out buffer (stored in gp-data) in case something contaminated it */
1819         gp_session_validatebuffer(p);
1820         
1821         /* set 'default' im2d_settings just in case something that uses this doesn't set it */
1822         p->im2d_settings.sizex= 1;
1823         p->im2d_settings.sizey= 1;
1824 }
1825
1826 /* cleanup after a painting session */
1827 static void gp_session_cleanup (tGPsdata *p)
1828 {
1829         bGPdata *gpd= p->gpd;
1830         
1831         /* error checking */
1832         if (gpd == NULL)
1833                 return;
1834         
1835         /* free stroke buffer */
1836         if (gpd->sbuffer) {
1837                 MEM_freeN(gpd->sbuffer);
1838                 gpd->sbuffer= NULL;
1839         }
1840         
1841         /* clear flags */
1842         gpd->sbuffer_size= 0;
1843         gpd->sbuffer_sflag= 0;
1844 }
1845
1846 /* init new stroke */
1847 static void gp_paint_initstroke (tGPsdata *p, short paintmode)
1848 {       
1849         /* get active layer (or add a new one if non-existent) */
1850         p->gpl= gpencil_layer_getactive(p->gpd);
1851         if (p->gpl == NULL)
1852                 p->gpl= gpencil_layer_addnew(p->gpd);
1853         if (p->gpl->flag & GP_LAYER_LOCKED) {
1854                 p->status= GP_STATUS_ERROR;
1855                 if (G.f & G_DEBUG)
1856                         printf("Error: Cannot paint on locked layer \n");
1857                 return;
1858         }
1859                 
1860         /* get active frame (add a new one if not matching frame) */
1861         p->gpf= gpencil_layer_getframe(p->gpl, p->scene->r.cfra, 1);
1862         if (p->gpf == NULL) {
1863                 p->status= GP_STATUS_ERROR;
1864                 if (G.f & G_DEBUG) 
1865                         printf("Error: No frame created (gpencil_paint_init) \n");
1866                 return;
1867         }
1868         else
1869                 p->gpf->flag |= GP_FRAME_PAINT;
1870         
1871         /* set 'eraser' for this stroke if using eraser */
1872         p->paintmode= paintmode;
1873         if (p->paintmode == GP_PAINTMODE_ERASER)
1874                 p->gpd->sbuffer_sflag |= GP_STROKE_ERASER;
1875         
1876         /* check if points will need to be made in view-aligned space */
1877         if (p->gpd->flag & GP_DATA_VIEWALIGN) {
1878                 switch (p->sa->spacetype) {
1879                         case SPACE_VIEW3D:
1880                         {
1881                                 View3D *v3d= (View3D *)p->sa->spacedata.first;
1882                                 RegionView3D *rv3d= NULL; // XXX
1883                                 float *fp= give_cursor(p->scene, v3d);
1884                                 
1885                                 initgrabz(rv3d, fp[0], fp[1], fp[2]);
1886                                 
1887                                 p->gpd->sbuffer_sflag |= GP_STROKE_3DSPACE;
1888                         }
1889                                 break;
1890                         case SPACE_NODE:
1891                         {
1892                                 p->gpd->sbuffer_sflag |= GP_STROKE_2DSPACE;
1893                         }
1894                                 break;
1895                         case SPACE_SEQ:
1896                         {
1897                                 SpaceSeq *sseq= (SpaceSeq *)p->sa->spacedata.first;
1898                                 int rectx, recty;
1899                                 float zoom, zoomx, zoomy;
1900                                 
1901                                 /* set draw 2d-stroke flag */
1902                                 p->gpd->sbuffer_sflag |= GP_STROKE_2DIMAGE;
1903                                 
1904                                 /* calculate zoom factor */
1905                                 zoom= (float)(SEQ_ZOOM_FAC(sseq->zoom));
1906                                 if (sseq->mainb == SEQ_DRAW_IMG_IMBUF) {
1907                                         zoomx = zoom * ((float)p->scene->r.xasp / (float)p->scene->r.yasp);
1908                                         zoomy = zoom;
1909                                 } 
1910                                 else
1911                                         zoomx = zoomy = zoom;
1912                                 
1913                                 /* calculate rect size to use to calculate the size of the drawing area
1914                                  *      - We use the size of the output image not the size of the ibuf being shown
1915                                  *        as it is too messy getting the ibuf (and could be too slow). This should be
1916                                  *        a reasonable for most cases anyway.
1917                                  */
1918                                 rectx= (p->scene->r.size * p->scene->r.xsch) / 100;
1919                                 recty= (p->scene->r.size * p->scene->r.ysch) / 100; 
1920                                 
1921                                 /* set offset and scale values for opertations to use */
1922                                 p->im2d_settings.sizex= (int)(zoomx * rectx);
1923                                 p->im2d_settings.sizey= (int)(zoomy * recty);
1924                                 p->im2d_settings.offsx= (int)((p->sa->winx-p->im2d_settings.sizex)/2 + sseq->xof);
1925                                 p->im2d_settings.offsy= (int)((p->sa->winy-p->im2d_settings.sizey)/2 + sseq->yof);
1926                         }
1927                                 break;
1928                         case SPACE_IMAGE:
1929                         {
1930                                 /* check if any ibuf available */
1931                                 if (p->ibuf)
1932                                         p->gpd->sbuffer_sflag |= GP_STROKE_2DSPACE;
1933                         }
1934                                 break;
1935                 }
1936         }
1937 }
1938
1939 /* finish off a stroke (clears buffer, but doesn't finish the paint operation) */
1940 static void gp_paint_strokeend (tGPsdata *p)
1941 {
1942         /* check if doing eraser or not */
1943         if ((p->gpd->sbuffer_sflag & GP_STROKE_ERASER) == 0) {
1944                 /* smooth stroke before transferring? */
1945                 gp_stroke_smooth(p);
1946                 
1947                 /* simplify stroke before transferring? */
1948                 gp_stroke_simplify(p);
1949                 
1950                 /* transfer stroke to frame */
1951                 gp_stroke_newfrombuffer(p);
1952         }
1953         
1954         /* clean up buffer now */
1955         gp_session_validatebuffer(p);
1956 }
1957
1958 /* finish off stroke painting operation */
1959 static void gp_paint_cleanup (tGPsdata *p)
1960 {
1961         /* finish off a stroke */
1962         gp_paint_strokeend(p);
1963         
1964         /* "unlock" frame */
1965         p->gpf->flag &= ~GP_FRAME_PAINT;
1966         
1967         /* add undo-push so stroke can be undone */
1968         /* FIXME: currently disabled, as it's impossible to get this working nice
1969          * as gpenci data is on currently screen-level (which isn't saved to undo files)
1970          */
1971         //BIF_undo_push("GPencil Stroke");
1972         
1973         /* force redraw after drawing action */
1974         // XXX force_draw_plus(SPACE_ACTION, 0);
1975 }
1976
1977 /* -------- */
1978
1979 /* main call to paint a new stroke */
1980 // XXX will become modal(), gets event, includes all info!
1981 short gpencil_paint (bContext *C, short paintmode)
1982 {
1983         tGPsdata p;
1984         short ok = GP_STROKEADD_NORMAL;
1985         
1986         /* init paint-data */
1987         gp_session_initpaint(C, &p);
1988         if (p.status == GP_STATUS_ERROR) {
1989                 gp_session_cleanup(&p);
1990                 return 0;
1991         }
1992         gp_paint_initstroke(&p, paintmode);
1993         if (p.status == GP_STATUS_ERROR) {
1994                 gp_session_cleanup(&p);
1995                 return 0;
1996         }
1997         
1998         /* set cursor to indicate drawing */
1999         // XXX (cursor callbacks in regiontype) setcursor_space(p.sa->spacetype, CURSOR_VPAINT);
2000         
2001         /* init drawing-device settings */
2002         // XXX getmouseco_areawin(p.mval);
2003         // XXX p.pressure = get_pressure();
2004         
2005         p.mvalo[0]= p.mval[0];
2006         p.mvalo[1]= p.mval[1];
2007         p.opressure= p.pressure;
2008         
2009         /* radius for eraser circle is defined in userprefs now */
2010         // TODO: make this more easily tweaked... 
2011         p.radius= U.gp_eraser;
2012         
2013         /* start drawing eraser-circle (if applicable) */
2014         //if (paintmode == GP_PAINTMODE_ERASER)
2015         // XXX  draw_sel_circle(p.mval, NULL, p.radius, p.radius, 0); // draws frontbuffer, but sets backbuf again
2016         
2017         /* only allow painting of single 'dots' if: 
2018          *      - pressure is not excessive (as it can be on some windows tablets)
2019          *      - draw-mode for active datablock is turned on
2020          *      - not erasing
2021          */
2022         if (paintmode != GP_PAINTMODE_ERASER) {
2023                 if (!(p.pressure >= 0.99f) || (p.gpd->flag & GP_DATA_EDITPAINT)) { 
2024                         gp_stroke_addpoint(&p, p.mval, p.pressure);
2025                 }
2026         }
2027         
2028         /* XXX paint loop */
2029         if(0) {
2030                 /* get current user input */
2031                 // XXX getmouseco_areawin(p.mval);
2032                 // XXX p.pressure = get_pressure();
2033                 
2034                 /* only add current point to buffer if mouse moved (otherwise wait until it does) */
2035                 if (paintmode == GP_PAINTMODE_ERASER) {
2036                         /* do 'live' erasing now */
2037                         gp_stroke_doeraser(&p);
2038                         
2039                         // XXX draw_sel_circle(p.mval, p.mvalo, p.radius, p.radius, 0);
2040                         // XXX force_draw(0);
2041                         
2042                         p.mvalo[0]= p.mval[0];
2043                         p.mvalo[1]= p.mval[1];
2044                         p.opressure= p.pressure;
2045                 }
2046                 else if (gp_stroke_filtermval(&p, p.mval, p.mvalo)) {
2047                         /* try to add point */
2048                         ok= gp_stroke_addpoint(&p, p.mval, p.pressure);
2049                         
2050                         /* handle errors while adding point */
2051                         if ((ok == GP_STROKEADD_FULL) || (ok == GP_STROKEADD_OVERFLOW)) {
2052                                 /* finish off old stroke */
2053                                 gp_paint_strokeend(&p);
2054                                 
2055                                 /* start a new stroke, starting from previous point */
2056                                 gp_stroke_addpoint(&p, p.mvalo, p.opressure);
2057                                 ok= gp_stroke_addpoint(&p, p.mval, p.pressure);
2058                         }
2059                         else if (ok == GP_STROKEADD_INVALID) {
2060                                 /* the painting operation cannot continue... */
2061                                 error("Cannot paint stroke");
2062                                 p.status = GP_STATUS_ERROR;
2063                                 
2064                                 if (G.f & G_DEBUG) 
2065                                         printf("Error: Grease-Pencil Paint - Add Point Invalid \n");
2066                                 // XXX break;
2067                         }
2068                         // XXX force_draw(0);
2069                         
2070                         p.mvalo[0]= p.mval[0];
2071                         p.mvalo[1]= p.mval[1];
2072                         p.opressure= p.pressure;
2073                 }
2074                 
2075                 /* do mouse checking at the end, so don't check twice, and potentially
2076                  * miss a short tap 
2077                  */
2078         }
2079         
2080         /* clear edit flags */
2081         G.f &= ~G_GREASEPENCIL;
2082         
2083         /* restore cursor to indicate end of drawing */
2084         // XXX  (cursor callbacks in regiontype) setcursor_space(p.sa->spacetype, CURSOR_STD);
2085         
2086         /* check size of buffer before cleanup, to determine if anything happened here */
2087         if (paintmode == GP_PAINTMODE_ERASER) {
2088                 ok= 1; /* assume that we did something... */
2089                 // XXX draw_sel_circle(NULL, p.mvalo, 0, p.radius, 0);
2090         }
2091         else
2092                 ok= p.gpd->sbuffer_size;
2093         
2094         /* cleanup */
2095         gp_paint_cleanup(&p);
2096         gp_session_cleanup(&p);
2097         
2098         /* done! return if a stroke was successfully added */
2099         return ok;
2100 }
2101
2102
2103 /* All event (loops) handling checking if stroke drawing should be initiated
2104  * should call this function.
2105  */
2106 short gpencil_do_paint (bContext *C)
2107 {
2108         ScrArea *sa= CTX_wm_area(C);
2109         bGPdata *gpd = gpencil_data_getactive(sa);
2110         short retval= 0;
2111         int alt= 0, shift= 0, mbut= 0; // XXX
2112         
2113         /* check if possible to do painting */
2114         if (gpd == NULL) 
2115                 return 0;
2116         
2117         /* currently, we will only 'paint' if:
2118          *      1. draw-mode on gpd is set (for accessibility reasons)
2119          *              a) single dots are only available by this method if a single click is made
2120          *              b) a straight line is drawn if ctrl-modifier is held (check is done when stroke is converted!)
2121          *      2. if shift-modifier is held + lmb -> 'quick paint'
2122          *
2123          *      OR
2124          * 
2125          * draw eraser stroke if:
2126          *      1. using the eraser on a tablet
2127          *      2. draw-mode on gpd is set (for accessiblity reasons)
2128          *              (eraser is mapped to right-mouse)
2129          *      3. Alt + 'select' mouse-button
2130          *              i.e.  if LMB = select: Alt-LMB
2131          *                        if RMB = select: Alt-RMB
2132          */
2133         if (get_activedevice() == 2) {
2134                 /* eraser on a tablet - always try to erase strokes */
2135                 retval = gpencil_paint(C, GP_PAINTMODE_ERASER);
2136         }
2137         else if (gpd->flag & GP_DATA_EDITPAINT) {
2138                 /* try to paint/erase */
2139                 if (mbut == L_MOUSE)
2140                         retval = gpencil_paint(C, GP_PAINTMODE_DRAW);
2141                 else if (mbut == R_MOUSE)
2142                         retval = gpencil_paint(C, GP_PAINTMODE_ERASER);
2143         }
2144         else if (!(gpd->flag & GP_DATA_LMBPLOCK)) {
2145                 /* try to paint/erase as not locked */
2146                 if (shift && (mbut == L_MOUSE)) {
2147                         retval = gpencil_paint(C, GP_PAINTMODE_DRAW);
2148                 }
2149                 else if (alt) {
2150                         if ((U.flag & USER_LMOUSESELECT) && (mbut == L_MOUSE))
2151                                 retval = gpencil_paint(C, GP_PAINTMODE_ERASER);
2152                         else if (!(U.flag & USER_LMOUSESELECT) && (mbut == R_MOUSE))
2153                                 retval = gpencil_paint(C, GP_PAINTMODE_ERASER);
2154                 }
2155         }
2156         
2157         /* return result of trying to paint */
2158         return retval;
2159 }
2160
2161 /* ************************************************** */