LCOV - code coverage report
Current view: top level - fontforgeexe - deltaui.c (source / functions) Hit Total Coverage
Test: FontForge coverage report 2017-08-04 01:21:11+02:00 (commit d35f7e4107a9e1db65cce47c468fcc914cecb8fd) Lines: 0 747 0.0 %
Date: 2017-08-04 Functions: 0 23 0.0 %

          Line data    Source code
       1             : /* Copyright (C) 2009-2012 by George Williams */
       2             : /*
       3             :  * Redistribution and use in source and binary forms, with or without
       4             :  * modification, are permitted provided that the following conditions are met:
       5             : 
       6             :  * Redistributions of source code must retain the above copyright notice, this
       7             :  * list of conditions and the following disclaimer.
       8             : 
       9             :  * Redistributions in binary form must reproduce the above copyright notice,
      10             :  * this list of conditions and the following disclaimer in the documentation
      11             :  * and/or other materials provided with the distribution.
      12             : 
      13             :  * The name of the author may not be used to endorse or promote products
      14             :  * derived from this software without specific prior written permission.
      15             : 
      16             :  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
      17             :  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
      18             :  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
      19             :  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
      20             :  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
      21             :  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
      22             :  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
      23             :  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
      24             :  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
      25             :  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
      26             :  */
      27             : #include "cvundoes.h"
      28             : #include "fontforgeui.h"
      29             : #include <ustring.h>
      30             : #include <chardata.h>
      31             : #include <utype.h>
      32             : #include <gkeysym.h>
      33             : #include <gimage.h>
      34             : #include <math.h>
      35             : #include "delta.h"
      36             : 
      37             : /* Suggestions for where delta instructions might be wanted */
      38             : 
      39             : #define CID_Sizes       100
      40             : #define CID_DPI         101
      41             : #define CID_BW          102
      42             : #define CID_Within      103
      43             : #define CID_Msg         104
      44             : #define CID_Ok          105
      45             : #define CID_Cancel      106
      46             : #define CID_Top         107
      47             : 
      48             : static double delta_within = .02;
      49             : static char *delta_sizes=NULL;
      50             : static int delta_dpi = 100;
      51             : static int delta_depth = 1;
      52             : 
      53             : static void StartDeltaDisplay(QGData *qg);
      54             : 
      55           0 : static int Delta_OK(GGadget *g, GEvent *e) {
      56           0 :     if ( e->type==et_controlevent && e->u.control.subtype == et_buttonactivate ) {
      57           0 :         QGData *qg = GDrawGetUserData(GGadgetGetWindow(g));
      58           0 :         int err=false;
      59             :         int dpi, depth;
      60             :         double within;
      61             :         char *sizes;
      62             : 
      63           0 :         within = GetReal8(qg->gw,CID_Within,_("Proximity"),&err);
      64           0 :         dpi = GetInt8(qg->gw,CID_DPI,_("DPI"),&err);
      65           0 :         if ( err )
      66           0 : return(true);
      67           0 :         if ( within<=0 || within>=.5 ) {
      68           0 :             ff_post_error(_("Bad Number"),_("The \"Proximity\" field must be more than 0 and less than a half."));
      69           0 : return( true );
      70             :         }
      71           0 :         if ( dpi<10 || dpi>5000 ) {
      72           0 :             ff_post_error(_("Unreasonable DPI"),_("The \"DPI\" field must be more than 10 and less than 5000."));
      73           0 : return( true );
      74             :         }
      75           0 :         depth = GGadgetIsChecked(GWidgetGetControl(qg->gw,CID_BW)) ? 1 : 8;
      76           0 :         sizes = GGadgetGetTitle8(GWidgetGetControl(qg->gw,CID_Sizes));
      77             : 
      78           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Msg),true);
      79           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Ok),false);
      80           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Cancel),false);
      81           0 :         GDrawSetCursor(qg->gw,ct_watch);
      82           0 :         GDrawProcessPendingEvents(NULL);
      83             : 
      84           0 :         qg->within = within;
      85           0 :         qg->dpi = dpi;
      86           0 :         qg->pixelsizes = sizes;
      87           0 :         qg->depth = depth;
      88           0 :         TopFindQuestionablePoints(qg);
      89             : 
      90           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Msg),false);
      91           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Ok),true);
      92           0 :         GGadgetSetVisible(GWidgetGetControl(qg->gw,CID_Cancel),true);
      93           0 :         GDrawSetCursor(qg->gw,ct_pointer);
      94           0 :         GDrawProcessPendingEvents(NULL);
      95             : 
      96           0 :         if ( qg->error!=qg_ok ) {
      97           0 :             switch ( qg->error ) {
      98             :               case qg_notnumber:
      99           0 :                 ff_post_error(_("Bad Number"),_("An entry in the \"Sizes\" field is not a number."));
     100           0 :               break;
     101             :               case qg_badnumber:
     102           0 :                 ff_post_error(_("Bad Number"),_("An entry in the \"Sizes\" field is unreasonable."));
     103           0 :               break;
     104             :               case qg_badrange:
     105           0 :                 ff_post_error(_("Bad Number"),_("An range in the \"Sizes\" field is incorrectly ordered."));
     106           0 :               break;
     107             :               case qg_nofont:
     108           0 :                 ff_post_error(_("FreeType unavailable"),_("FreeType unavailable."));
     109           0 :               break;
     110             :               default:
     111           0 :                 IError(_("Unexpected error"));
     112           0 :               break;
     113             :             }
     114           0 :             free(sizes);
     115           0 :             qg->cur = 0;
     116           0 : return( true );
     117             :         }
     118             : 
     119           0 :         free(delta_sizes);
     120           0 :         delta_within = within;
     121           0 :         delta_dpi    = dpi;
     122           0 :         delta_depth  = depth;
     123           0 :         delta_sizes  = sizes;
     124             : 
     125           0 :         if ( qg->cur==0 ) {
     126           0 :             ff_post_error(_("Nothing found"),_("Nothng found."));
     127           0 :             qg->done = true;
     128           0 : return( true );
     129             :         }
     130             : 
     131           0 :         if ( qg->cur >= qg->max )
     132           0 :             qg->qg = realloc(qg->qg,(qg->max += 1) * sizeof(QuestionableGrid));
     133           0 :         memset(qg->qg+qg->cur,0,sizeof(QuestionableGrid));
     134           0 :         GDrawSetVisible(qg->gw,false);
     135           0 :         StartDeltaDisplay(qg);
     136           0 :         qg->done = true;
     137             :     }
     138           0 : return( true );
     139             : }
     140             : 
     141           0 : static int Delta_Cancel(GGadget *g, GEvent *e) {
     142           0 :     if ( e->type==et_controlevent && e->u.control.subtype == et_buttonactivate ) {
     143           0 :         QGData *qg = GDrawGetUserData(GGadgetGetWindow(g));
     144           0 :         qg->done = true;
     145             :     }
     146           0 : return( true );
     147             : }
     148             : 
     149           0 : static int delta_e_h(GWindow gw, GEvent *event) {
     150           0 :     if ( event->type==et_close ) {
     151           0 :         QGData *qg = GDrawGetUserData(gw);
     152           0 :         qg->done = true;
     153           0 :     } else if ( event->type == et_char ) {
     154           0 : return( false );
     155           0 :     } else if ( event->type == et_map ) {
     156             :         /* Above palettes */
     157           0 :         GDrawRaise(gw);
     158           0 :     } else if ( event->type == et_destroy ) {
     159           0 :         QGData *qg = GDrawGetUserData(gw);
     160           0 :         free(qg->qg);
     161           0 :         free(qg);
     162             :     }
     163           0 : return( true );
     164             : }
     165             : 
     166           0 : void DeltaSuggestionDlg(FontView *fv,CharView *cv) {
     167             :     GRect pos;
     168             :     GWindow gw;
     169             :     GWindowAttrs wattrs;
     170             :     GGadgetCreateData gcd[13], boxes[4];
     171             :     GTextInfo label[13];
     172             :     GGadgetCreateData *varray[7][5], *barray[9];
     173             :     char dpi_buffer[40], within_buffer[40];
     174             :     QGData *data;
     175           0 :     int failed = false;
     176             :     int k, r;
     177             :     FontView *savefv;
     178             : 
     179           0 :     if ( !hasFreeType() ) {
     180           0 :         ff_post_error(_("No FreeType"),_("You must install the freetype library before using this command."));
     181           0 : return;
     182             :     }
     183           0 :     if ( !hasFreeTypeByteCode() ) {
     184           0 :         ff_post_error(_("No FreeType"),_("Your version of the freetype library does not contain the bytecode interpreter."));
     185           0 : return;
     186             :     }
     187             :     
     188           0 :     if ( delta_sizes==NULL )
     189           0 :         delta_sizes = copy("7-40,72,80,88,96");
     190             : 
     191           0 :     data = calloc(1,sizeof(QGData));
     192           0 :     data->fv = (FontViewBase *) fv;
     193           0 :     data->cv = cv;
     194           0 :     if ( cv!=NULL ) {
     195           0 :         data->sc = cv->b.sc;
     196           0 :         savefv = (FontView *) cv->b.fv;
     197           0 :         data->layer = CVLayer((CharViewBase *) cv);
     198           0 :         if ( !data->sc->parent->layers[data->layer].order2 )
     199           0 :             failed = true;
     200           0 :         if ( !failed && data->sc->ttf_instrs_len==0 )
     201           0 :             ff_post_notice(_("No Instructions"),_("This glyph has no instructions. Adding instructions (a DELTA) may change its rasterization significantly."));
     202           0 :         cv->qg = data;
     203           0 :         cv->note_x = cv->note_y = 32766;
     204             :     } else {
     205           0 :         savefv = fv;
     206           0 :         data->layer = fv->b.active_layer;
     207           0 :         if ( !fv->b.sf->layers[data->layer].order2 )
     208           0 :             failed = true;
     209             :     }
     210           0 :     if ( failed ) {
     211           0 :         ff_post_error(_("Not quadratic"),_("This must be a truetype layer."));
     212           0 :         free(data);
     213           0 :         if ( cv!=NULL )
     214           0 :             cv->qg = NULL;
     215           0 : return;
     216             :     }
     217           0 :     savefv->qg = data;
     218             : 
     219           0 :     memset(&wattrs,0,sizeof(wattrs));
     220           0 :     wattrs.mask = wam_events|wam_cursor|wam_utf8_wtitle|wam_undercursor|wam_isdlg|wam_restrict;
     221           0 :     wattrs.event_masks = ~(1<<et_charup);
     222           0 :     wattrs.restrict_input_to_me = 1;
     223           0 :     wattrs.undercursor = 1;
     224           0 :     wattrs.cursor = ct_pointer;
     225           0 :     wattrs.utf8_window_title = _("DELTA suggestions");
     226           0 :     wattrs.is_dlg = true;
     227           0 :     pos.x = pos.y = 0;
     228           0 :     pos.width = GGadgetScale(GDrawPointsToPixels(NULL,190));
     229           0 :     pos.height = GDrawPointsToPixels(NULL,106);
     230           0 :     data->gw = gw = GDrawCreateTopWindow(NULL,&pos,delta_e_h,data,&wattrs);
     231             : 
     232           0 :     memset(&label,0,sizeof(label));
     233           0 :     memset(&gcd,0,sizeof(gcd));
     234           0 :     memset(&boxes,0,sizeof(boxes));
     235             : 
     236           0 :     k=r=0;
     237             : 
     238           0 :     label[k].text = (unichar_t *) _(
     239             :             "When a curve passes very close to the center of a\n"
     240             :             "pixel you might want to check that the curve is on\n"
     241             :             "the intended side of that pixel.\n"
     242             :             "If it's on the wrong side, consider using a DELTA\n"
     243             :             "instruction to adjust the closest point at the\n"
     244             :             "current pixelsize."
     245             :         );
     246           0 :     label[k].text_is_1byte = true;
     247           0 :     label[k].text_in_resource = true;
     248           0 :     gcd[k].gd.label = &label[k];
     249           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     250           0 :     gcd[k++].creator = GLabelCreate;
     251           0 :     varray[r][0] = &gcd[k-1]; varray[r][1] = GCD_ColSpan; varray[r][2] = GCD_ColSpan; varray[r][3] = GCD_ColSpan; varray[r++][4] = NULL;
     252             : 
     253           0 :     label[k].text = (unichar_t *) _("Rasterize at sizes:");
     254           0 :     label[k].text_is_1byte = true;
     255           0 :     label[k].text_in_resource = true;
     256           0 :     gcd[k].gd.label = &label[k];
     257           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     258           0 :     gcd[k++].creator = GLabelCreate;
     259             : 
     260           0 :     label[k].text = (unichar_t *) delta_sizes;
     261           0 :     label[k].text_is_1byte = true;
     262           0 :     label[k].text_in_resource = true;
     263           0 :     gcd[k].gd.label = &label[k];
     264           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     265           0 :     gcd[k].gd.cid = CID_Sizes;
     266           0 :     gcd[k++].creator = GTextFieldCreate;
     267           0 :     varray[r][0] = &gcd[k-2]; varray[r][1] = &gcd[k-1]; varray[r][2] = GCD_ColSpan; varray[r][3] = GCD_ColSpan; varray[r++][4] = NULL;
     268             : 
     269           0 :     label[k].text = (unichar_t *) _("DPI:");
     270           0 :     label[k].text_is_1byte = true;
     271           0 :     label[k].text_in_resource = true;
     272           0 :     gcd[k].gd.label = &label[k];
     273           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     274           0 :     gcd[k++].creator = GLabelCreate;
     275             : 
     276           0 :     sprintf( dpi_buffer, "%d", delta_dpi );
     277           0 :     label[k].text = (unichar_t *) dpi_buffer;
     278           0 :     label[k].text_is_1byte = true;
     279           0 :     label[k].text_in_resource = true;
     280           0 :     gcd[k].gd.label = &label[k];
     281           0 :     gcd[k].gd.pos.width = 40;
     282           0 :     gcd[k].gd.cid = CID_DPI;
     283           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     284           0 :     gcd[k++].creator = GTextFieldCreate;
     285             : 
     286           0 :     label[k].text = (unichar_t *) _("_Mono");
     287           0 :     label[k].text_is_1byte = true;
     288           0 :     label[k].text_in_resource = true;
     289           0 :     gcd[k].gd.label = &label[k];
     290           0 :     gcd[k].gd.flags = delta_depth==1 ? (gg_enabled|gg_visible|gg_cb_on) : (gg_enabled|gg_visible);
     291           0 :     gcd[k].gd.cid = CID_BW;
     292           0 :     gcd[k++].creator = GRadioCreate;
     293           0 :     varray[r][0] = &gcd[k-1]; varray[r][1] = GCD_HPad10;
     294             : 
     295           0 :     label[k].text = (unichar_t *) _("_Anti-Aliased");
     296           0 :     label[k].text_is_1byte = true;
     297           0 :     label[k].text_in_resource = true;
     298           0 :     gcd[k].gd.label = &label[k];
     299           0 :     gcd[k].gd.flags = delta_depth!=1 ? (gg_enabled|gg_visible|gg_cb_on) : (gg_enabled|gg_visible);
     300           0 :     gcd[k++].creator = GRadioCreate;
     301           0 :     varray[r][0] = &gcd[k-4]; varray[r][1] = &gcd[k-3]; varray[r][2] = &gcd[k-2]; varray[r][3] = &gcd[k-1]; varray[r++][4] = NULL;
     302             : 
     303           0 :     label[k].text = (unichar_t *) _("Proximity:");
     304           0 :     label[k].text_is_1byte = true;
     305           0 :     label[k].text_in_resource = true;
     306           0 :     gcd[k].gd.label = &label[k];
     307           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     308           0 :     gcd[k++].creator = GLabelCreate;
     309             : 
     310           0 :     sprintf( within_buffer, "%g", delta_within );
     311           0 :     label[k].text = (unichar_t *) within_buffer;
     312           0 :     label[k].text_is_1byte = true;
     313           0 :     label[k].text_in_resource = true;
     314           0 :     gcd[k].gd.label = &label[k];
     315           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     316           0 :     gcd[k].gd.pos.width = 40;
     317           0 :     gcd[k].gd.cid = CID_Within;
     318           0 :     gcd[k++].creator = GTextFieldCreate;
     319             : 
     320           0 :     label[k].text = (unichar_t *) _("pixels");
     321           0 :     label[k].text_is_1byte = true;
     322           0 :     label[k].text_in_resource = true;
     323           0 :     gcd[k].gd.label = &label[k];
     324           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
     325           0 :     gcd[k++].creator = GLabelCreate;
     326           0 :     varray[r][0] = &gcd[k-3]; varray[r][1] = &gcd[k-2]; varray[r][2] = &gcd[k-1]; varray[r][3] = GCD_ColSpan; varray[r++][4] = NULL;
     327             : 
     328           0 :     label[k].text = (unichar_t *) _( "This may take a while. Please be patient..." );
     329           0 :     label[k].text_is_1byte = true;
     330           0 :     label[k].text_in_resource = true;
     331           0 :     gcd[k].gd.label = &label[k];
     332           0 :     gcd[k].gd.flags = gg_enabled;
     333           0 :     gcd[k].gd.cid = CID_Msg;
     334           0 :     gcd[k++].creator = GLabelCreate;
     335           0 :     varray[r][0] = &gcd[k-1]; varray[r][1] = GCD_ColSpan; varray[r][2] = GCD_ColSpan; varray[r][3] = GCD_ColSpan; varray[r++][4] = NULL;
     336             : 
     337           0 :     gcd[k].gd.flags = gg_visible | gg_enabled | gg_but_default;
     338           0 :     label[k].text = (unichar_t *) _("_OK");
     339           0 :     label[k].text_is_1byte = true;
     340           0 :     label[k].text_in_resource = true;
     341           0 :     gcd[k].gd.label = &label[k];
     342           0 :     gcd[k].gd.handle_controlevent = Delta_OK;
     343           0 :     gcd[k].gd.cid = CID_Ok;
     344           0 :     gcd[k++].creator = GButtonCreate;
     345           0 :     barray[0] = GCD_Glue; barray[1] = &gcd[k-1]; barray[2] = GCD_Glue;
     346             : 
     347           0 :     gcd[k].gd.flags = gg_visible | gg_enabled | gg_but_cancel;
     348           0 :     label[k].text = (unichar_t *) _("_Cancel");
     349           0 :     label[k].text_is_1byte = true;
     350           0 :     label[k].text_in_resource = true;
     351           0 :     gcd[k].gd.label = &label[k];
     352           0 :     gcd[k].gd.handle_controlevent = Delta_Cancel;
     353           0 :     gcd[k].gd.cid = CID_Cancel;
     354           0 :     gcd[k++].creator = GButtonCreate;
     355           0 :     barray[3] = GCD_Glue; barray[4] = &gcd[k-1]; barray[5] = GCD_Glue; barray[6] = NULL;
     356             : 
     357           0 :     boxes[2].gd.flags = gg_enabled|gg_visible;
     358           0 :     boxes[2].gd.u.boxelements = barray;
     359           0 :     boxes[2].creator = GHBoxCreate;
     360           0 :     varray[r][0] = &boxes[2]; varray[r][1] = GCD_ColSpan; varray[r][2] = GCD_ColSpan; varray[r][3] = GCD_ColSpan; varray[r++][4] = NULL;
     361           0 :     varray[r][0] = NULL;
     362             : 
     363           0 :     boxes[0].gd.pos.x = boxes[0].gd.pos.y = 2;
     364           0 :     boxes[0].gd.flags = gg_enabled|gg_visible;
     365           0 :     boxes[0].gd.u.boxelements = varray[0];
     366           0 :     boxes[0].gd.cid = CID_Top;
     367           0 :     boxes[0].creator = GHVGroupCreate;
     368             : 
     369             : 
     370           0 :     GGadgetsCreate(gw,boxes);
     371           0 :     GHVBoxFitWindow(boxes[0].ret);
     372             : 
     373           0 :     GDrawSetVisible(gw,true);
     374           0 :     while ( !data->done )
     375           0 :         GDrawProcessOneEvent(NULL);
     376           0 :     GDrawDestroyWindow(gw);
     377           0 :     if ( data->cv!=NULL )
     378           0 :         data->cv->qg = NULL;
     379           0 :     if ( savefv->qg == data )
     380           0 :         savefv->qg = NULL;
     381             : }
     382             : 
     383           0 : void QGRmCharView(QGData *qg,CharView *cv) {
     384           0 :     if ( qg->cv==cv )
     385           0 :         qg->cv = NULL;
     386           0 : }
     387             : 
     388           0 : void QGRmFontView(QGData *qg,FontView *fv) {
     389           0 :     qg->done = true;
     390           0 :     fv->qg = NULL;
     391           0 : }
     392             : /* ************************************************************************** */
     393             : /* ************************************************************************** */
     394             : 
     395             : #define CID_Sort        200
     396             : #define CID_GlyphSort   201
     397             : 
     398             : static GTextInfo sorts[] = {
     399             :     { (unichar_t *) N_("Glyph, Size, Point"), NULL, 0, 0, (void *) is_glyph_size_pt, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     400             :     { (unichar_t *) N_("Glyph, Point, Size"), NULL, 0, 0, (void *) is_glyph_pt_size, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     401             :     { (unichar_t *) N_("Size, Glyph, Point"), NULL, 0, 0, (void *) is_size_glyph_pt, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     402             :     GTEXTINFO_EMPTY
     403             : };
     404             : static GTextInfo glyphsorts[] = {
     405             :     { (unichar_t *) N_("Unicode"), NULL, 0, 0, (void *) gs_unicode, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     406             :     { (unichar_t *) N_("Sort|Alphabetic"), NULL, 0, 0, (void *) gs_alpha, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     407             :     { (unichar_t *) N_("Glyph Order"), NULL, 0, 0, (void *) gs_gid, NULL, 0, 0, 0, 0, 0, 0, 1, 0, 0, '\0' },
     408             :     GTEXTINFO_EMPTY
     409             : };
     410             : 
     411           0 : static int QG_VScroll(GGadget *g, GEvent *e) {
     412           0 :     QGData *qg = GDrawGetUserData(GGadgetGetWindow(g));
     413           0 :     int newpos = qg->loff_top;
     414             : 
     415           0 :     switch( e->u.control.u.sb.type ) {
     416             :       case et_sb_top:
     417           0 :         newpos = 0;
     418           0 :       break;
     419             :       case et_sb_uppage:
     420           0 :         newpos -= 9*qg->vlcnt/10;
     421           0 :       break;
     422             :       case et_sb_up:
     423           0 :         newpos -= qg->vlcnt/15;
     424           0 :       break;
     425             :       case et_sb_down:
     426           0 :         newpos += qg->vlcnt/15;
     427           0 :       break;
     428             :       case et_sb_downpage:
     429           0 :         newpos += 9*qg->vlcnt/10;
     430           0 :       break;
     431             :       case et_sb_bottom:
     432           0 :         newpos = 0;
     433           0 :       break;
     434             :       case et_sb_thumb:
     435             :       case et_sb_thumbrelease:
     436           0 :         newpos = e->u.control.u.sb.pos;
     437           0 :       break;
     438             :       case et_sb_halfup:
     439           0 :         newpos -= qg->vlcnt/30;
     440           0 :       break;
     441             :       case et_sb_halfdown:
     442           0 :         newpos += qg->vlcnt/30;
     443           0 :       break;
     444             :     }
     445           0 :     if ( newpos + qg->vlcnt > qg->lcnt )
     446           0 :         newpos = qg->lcnt-qg->vlcnt;
     447           0 :     if ( newpos<0 )
     448           0 :         newpos = 0;
     449           0 :     if ( qg->loff_top!=newpos ) {
     450           0 :         qg->loff_top = newpos;
     451           0 :         GScrollBarSetPos(qg->vsb,newpos);
     452           0 :         GDrawRequestExpose(qg->v,NULL,false);
     453             :     }
     454           0 : return( true );
     455             : }
     456             : 
     457           0 : static void QG_SetSb(QGData *qg) {
     458           0 :     if ( qg->loff_top + qg->vlcnt > qg->lcnt )
     459           0 :         qg->loff_top = qg->lcnt-qg->vlcnt;
     460           0 :     if ( qg->loff_top<0 )
     461           0 :         qg->loff_top = 0;
     462           0 :     GScrollBarSetBounds(qg->vsb,0,qg->lcnt,qg->vlcnt);
     463           0 :     GScrollBarSetPos(qg->vsb,qg->loff_top);
     464           0 : }
     465             : 
     466           0 : static int QG_Count(struct qgnode *parent) {
     467           0 :     int l, cnt=0;
     468             : 
     469           0 :     if ( !parent->open )
     470           0 : return( 0 );
     471           0 :     if ( parent->kids==NULL )
     472           0 : return( parent->qg_cnt );
     473             : 
     474           0 :     for ( l=0; l<parent->kid_cnt; ++l ) {
     475           0 :         parent->kids[l].tot_under = QG_Count(&parent->kids[l]);
     476           0 :         cnt += 1 + parent->kids[l].tot_under;
     477             :     }
     478           0 : return( cnt );
     479             : }
     480             : 
     481           0 : static void QG_Remetric(QGData *qg) {
     482             :     GRect size;
     483             : 
     484           0 :     GDrawGetSize(qg->v,&size);
     485           0 :     qg->vlcnt = size.height/qg->fh;
     486           0 :     qg->lcnt = QG_Count(&qg->list);
     487           0 :     QG_SetSb(qg);
     488           0 : }
     489             : 
     490             : struct navigate {
     491             :     struct qgnode *parent;
     492             :     int offset;                 /* offset of -1 means the qgnode */
     493             : };
     494             : 
     495           0 : static void QG_FindLine(struct qgnode *parent, int l,struct navigate *where) {
     496             :     int k;
     497             : 
     498           0 :     if ( parent->kids == NULL ) {
     499           0 :         if ( l<parent->qg_cnt ) {
     500           0 :             where->parent = parent;
     501           0 :             where->offset = l;
     502             :         } else {
     503           0 :             where->parent = NULL;
     504           0 :             where->offset = -1;
     505             :         }
     506           0 : return;
     507             :     }
     508           0 :     for ( k=0; k<parent->kid_cnt; ++k ) {
     509           0 :         if ( l==0 ) {
     510           0 :             where->parent = &parent->kids[k];
     511           0 :             where->offset = -1;
     512           0 : return;
     513             :         }
     514           0 :         --l;
     515           0 :         if ( parent->kids[k].open ) {
     516           0 :             if ( l<parent->kids[k].tot_under ) {
     517           0 :                 QG_FindLine(&parent->kids[k],l,where);
     518           0 : return;
     519             :             } else
     520           0 :                 l -= parent->kids[k].tot_under;
     521             :         }
     522             :     }
     523           0 :     where->parent = NULL;
     524           0 :     where->offset = -1;
     525             : }
     526             : 
     527           0 : static void QG_NextLine(struct navigate *where) {
     528             :     int k;
     529             : 
     530           0 :     if ( where->parent==NULL )
     531           0 : return;
     532           0 :     if ( where->offset!=-1 && where->offset+1<where->parent->qg_cnt ) {
     533           0 :         ++where->offset;
     534           0 : return;
     535             :     }
     536           0 :     if ( where->parent->open && where->offset==-1 ) {
     537           0 :         if ( where->parent->kids==NULL ) {
     538           0 :             where->offset = 0;
     539           0 : return;
     540             :         } else {
     541           0 :             where->parent = &where->parent->kids[0];
     542           0 :             where->offset = -1;
     543           0 : return;
     544             :         }
     545             :     }
     546             :     for (;;) {
     547           0 :         if ( where->parent->parent==NULL ) {
     548           0 :             where->parent = NULL;
     549           0 :             where->offset = -1;
     550           0 : return;
     551             :         }
     552           0 :         k = where->parent - where->parent->parent->kids;
     553           0 :         if ( k+1< where->parent->parent->kid_cnt ) {
     554           0 :             ++where->parent;
     555           0 :             where->offset = -1;
     556           0 : return;
     557             :         }
     558           0 :         where->parent = where->parent->parent;
     559           0 :     }
     560             : }
     561             : 
     562           0 : static void qgnodeFree(struct qgnode *parent) {
     563             :     int i;
     564             : 
     565           0 :     for ( i=0; i<parent->kid_cnt; ++i )
     566           0 :         qgnodeFree(&parent->kids[i]);
     567           0 :     free(parent->kids);
     568           0 :     free(parent->name);
     569           0 : }
     570             : 
     571             : static const QGData *kludge;
     572             : 
     573           0 : static int qg_sorter(const void *pt1, const void *pt2) {
     574           0 :     const QuestionableGrid *q1 = pt1, *q2 = pt2;
     575           0 :     const QGData *qg = kludge;
     576           0 :     int pt_cmp = q1->nearestpt - q2->nearestpt;
     577           0 :     int size_cmp = q1->size - q2->size;
     578             :     int g_cmp;
     579             :     int t1, t2, t3;
     580             : 
     581           0 :     switch ( qg->glyph_sort ) {
     582             :       case gs_unicode:
     583           0 :         g_cmp = q1->sc->unicodeenc - q2->sc->unicodeenc;
     584           0 :       break;
     585             :       case gs_gid:
     586           0 :         g_cmp = q1->sc->orig_pos - q2->sc->orig_pos;
     587           0 :       break;
     588             :       case gs_alpha:
     589           0 :         g_cmp = strcmp(q1->sc->name,q2->sc->name);
     590           0 :       break;
     591             :     }
     592           0 :     switch ( qg->info_sort ) {
     593             :       case is_glyph_size_pt:
     594           0 :         t1 = g_cmp; t2 = size_cmp; t3 = pt_cmp;
     595           0 :       break;
     596             :       case is_glyph_pt_size:
     597           0 :         t1 = g_cmp; t2 = pt_cmp; t3 = size_cmp;
     598           0 :       break;
     599             :       case is_size_glyph_pt:
     600           0 :         t1 = size_cmp; t2 = g_cmp; t3 = pt_cmp;
     601           0 :       break;
     602             :     }
     603           0 :     if ( t1!=0 )
     604           0 : return( t1 );
     605           0 :     if ( t2!=0 )
     606           0 : return( t2 );
     607             : 
     608           0 : return( t3 );
     609             : }
     610             : 
     611           0 : static void QGSecondLevel(QGData *qg, struct qgnode *parent) {
     612             :     int cnt, l, lstart;
     613             :     int size;
     614             :     SplineChar *sc;
     615             :     int pt;
     616             :     char buffer[200];
     617             : 
     618           0 :     switch ( qg->info_sort ) {
     619             :       case is_glyph_size_pt:    /* size */
     620           0 :         size = -1;
     621           0 :         cnt = 0;
     622           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     623           0 :             if ( size!=parent->first[l].size ) {
     624           0 :                 ++cnt;
     625           0 :                 size = parent->first[l].size;
     626             :             }
     627             :         }
     628           0 :         parent->kid_cnt = cnt;
     629           0 :         parent->kids = calloc(cnt,sizeof(struct qgnode));
     630           0 :         cnt = 0;
     631           0 :         lstart = 0; size=-1;
     632           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     633           0 :             if ( size!=parent->first[l].size && size!=-1 ) {
     634           0 :                 sprintf( buffer, _("Size: %d (%d)"), size, l-lstart );
     635           0 :                 parent->kids[cnt].name = copy(buffer);
     636           0 :                 parent->kids[cnt].parent = parent;
     637           0 :                 parent->kids[cnt].first = &parent->first[lstart];
     638           0 :                 parent->kids[cnt].qg_cnt = l-lstart;
     639           0 :                 ++cnt;
     640           0 :                 lstart = l;
     641           0 :                 size = parent->first[l].size;
     642           0 :             } else if ( size!=parent->first[l].size )
     643           0 :                 size = parent->first[l].size;
     644             :         }
     645           0 :         if ( size!=-1 ) {
     646           0 :             sprintf( buffer, _("Size: %d (%d)"), size, l-lstart );
     647           0 :             parent->kids[cnt].name = copy(buffer);
     648           0 :             parent->kids[cnt].parent = parent;
     649           0 :             parent->kids[cnt].first = &parent->first[lstart];
     650           0 :             parent->kids[cnt].qg_cnt = l-lstart;
     651             :         }
     652           0 :       break;
     653             :       case is_glyph_pt_size:    /* pt */
     654           0 :         pt = -1;
     655           0 :         cnt = 0;
     656           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     657           0 :             if ( pt!=parent->first[l].nearestpt ) {
     658           0 :                 ++cnt;
     659           0 :                 pt = parent->first[l].nearestpt;
     660             :             }
     661             :         }
     662           0 :         parent->kid_cnt = cnt;
     663           0 :         parent->kids = calloc(cnt,sizeof(struct qgnode));
     664           0 :         cnt = 0;
     665           0 :         lstart = 0; pt=-1;
     666           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     667           0 :             if ( pt!=parent->first[l].nearestpt && pt!=-1 ) {
     668           0 :                 sprintf( buffer, _("Point: %d (%d)"), pt, l-lstart );
     669           0 :                 parent->kids[cnt].name = copy(buffer);
     670           0 :                 parent->kids[cnt].parent = parent;
     671           0 :                 parent->kids[cnt].first = &parent->first[lstart];
     672           0 :                 parent->kids[cnt].qg_cnt = l-lstart;
     673           0 :                 ++cnt;
     674           0 :                 lstart = l;
     675           0 :                 pt = parent->first[l].nearestpt;
     676           0 :             } else if ( pt!=parent->first[l].nearestpt )
     677           0 :                 pt = parent->first[l].nearestpt;
     678             :         }
     679           0 :         if ( pt!=-1 ) {
     680           0 :             sprintf( buffer, _("Point: %d (%d)"), pt, l-lstart );
     681           0 :             parent->kids[cnt].name = copy(buffer);
     682           0 :             parent->kids[cnt].parent = parent;
     683           0 :             parent->kids[cnt].first = &parent->first[lstart];
     684           0 :             parent->kids[cnt].qg_cnt = l-lstart;
     685             :         }
     686           0 :       break;
     687             :       case is_size_glyph_pt:    /* glyph */
     688           0 :         sc = NULL;
     689           0 :         cnt = 0;
     690           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     691           0 :             if ( sc!=parent->first[l].sc ) {
     692           0 :                 ++cnt;
     693           0 :                 sc = parent->first[l].sc;
     694             :             }
     695             :         }
     696           0 :         parent->kid_cnt = cnt;
     697           0 :         parent->kids = calloc(cnt,sizeof(struct qgnode));
     698           0 :         cnt = 0;
     699           0 :         lstart = 0;
     700           0 :         sc = NULL;
     701           0 :         for ( l=0; l<parent->qg_cnt; ++l ) {
     702           0 :             if ( sc!=parent->first[l].sc && sc!=NULL ) {
     703           0 :                 sprintf( buffer, "\"%.40s\" (%d)", sc->name, l-lstart );
     704           0 :                 parent->kids[cnt].name = copy(buffer);
     705           0 :                 parent->kids[cnt].parent = parent;
     706           0 :                 parent->kids[cnt].first = &parent->first[lstart];
     707           0 :                 parent->kids[cnt].qg_cnt = l-lstart;
     708           0 :                 ++cnt;
     709           0 :                 lstart = l;
     710           0 :                 sc = parent->first[l].sc;
     711           0 :             } else if ( sc!=parent->first[l].sc )
     712           0 :                 sc = parent->first[l].sc;
     713             :         }
     714           0 :         if ( sc!=NULL ) {
     715           0 :             sprintf( buffer, "\"%.40s\" (%d)", sc->name, l-lstart );
     716           0 :             parent->kids[cnt].name = copy(buffer);
     717           0 :             parent->kids[cnt].parent = parent;
     718           0 :             parent->kids[cnt].first = &parent->first[lstart];
     719           0 :             parent->kids[cnt].qg_cnt = l-lstart;
     720             :         }
     721           0 :       break;
     722             :     }
     723           0 : }
     724             : 
     725           0 : static void QGDoSort(QGData *qg) {
     726             :     int pos, l, k, cnt, lstart;
     727             :     char buffer[200];
     728             : 
     729           0 :     pos = GGadgetGetFirstListSelectedItem(GWidgetGetControl(qg->gw,CID_Sort));
     730           0 :     qg->info_sort = (intpt) sorts[pos].userdata;
     731             : 
     732           0 :     pos = GGadgetGetFirstListSelectedItem(GWidgetGetControl(qg->gw,CID_GlyphSort));
     733           0 :     qg->glyph_sort = (intpt) glyphsorts[pos].userdata;
     734             : 
     735           0 :     kludge = qg;
     736           0 :     qsort(qg->qg,qg->cur,sizeof(QuestionableGrid),qg_sorter);
     737             : 
     738           0 :     qgnodeFree(&qg->list);
     739           0 :     memset(&qg->list,0,sizeof(struct qgnode));
     740           0 :     qg->list.open = true;
     741           0 :     qg->list.first = qg->qg;
     742           0 :     if ( qg->info_sort == is_glyph_size_pt || qg->info_sort == is_glyph_pt_size ) {
     743           0 :         SplineChar *sc = NULL;
     744           0 :         cnt = 0;
     745           0 :         for ( l=0; l<qg->cur; ++l ) {
     746           0 :             if ( sc!=qg->qg[l].sc ) {
     747           0 :                 ++cnt;
     748           0 :                 sc = qg->qg[l].sc;
     749             :             }
     750             :         }
     751           0 :         qg->list.kid_cnt = cnt;
     752           0 :         qg->list.kids = calloc(cnt,sizeof(struct qgnode));
     753           0 :         cnt = 0;
     754           0 :         lstart = 0; sc=NULL;
     755           0 :         for ( l=0; l<qg->cur; ++l ) {
     756           0 :             if ( sc!=qg->qg[l].sc && sc!=NULL ) {
     757           0 :                 sprintf( buffer, "\"%.40s\" (%d)", sc->name, l-lstart );
     758           0 :                 qg->list.kids[cnt].name = copy(buffer);
     759           0 :                 qg->list.kids[cnt].parent = &qg->list;
     760           0 :                 qg->list.kids[cnt].first = &qg->qg[lstart];
     761           0 :                 qg->list.kids[cnt].qg_cnt = l-lstart;
     762           0 :                 ++cnt;
     763           0 :                 lstart = l;
     764           0 :                 sc = qg->qg[l].sc;
     765           0 :             } else if ( sc!=qg->qg[l].sc )
     766           0 :                 sc = qg->qg[l].sc;
     767             :         }
     768           0 :         if ( sc!=NULL ) {
     769           0 :             sprintf( buffer, "\"%.40s\" (%d)", sc->name, l-lstart );
     770           0 :             qg->list.kids[cnt].name = copy(buffer);
     771           0 :             qg->list.kids[cnt].parent = &qg->list;
     772           0 :             qg->list.kids[cnt].first = &qg->qg[lstart];
     773           0 :             qg->list.kids[cnt].qg_cnt = l-lstart;
     774             :         }
     775             :     } else {
     776           0 :         int size = -1;
     777           0 :         cnt = 0;
     778           0 :         for ( l=0; l<qg->cur; ++l ) {
     779           0 :             if ( size!=qg->qg[l].size ) {
     780           0 :                 ++cnt;
     781           0 :                 size = qg->qg[l].size;
     782             :             }
     783             :         }
     784           0 :         qg->list.kid_cnt = cnt;
     785           0 :         qg->list.kids = calloc(cnt,sizeof(struct qgnode));
     786           0 :         cnt = 0;
     787           0 :         lstart = 0; size=-1;
     788           0 :         for ( l=0; l<qg->cur; ++l ) {
     789           0 :             if ( size!=qg->qg[l].size && size!=-1 ) {
     790           0 :                 sprintf( buffer, _("Size: %d (%d)"), size, l-lstart );
     791           0 :                 qg->list.kids[cnt].name = copy(buffer);
     792           0 :                 qg->list.kids[cnt].parent = &qg->list;
     793           0 :                 qg->list.kids[cnt].first = &qg->qg[lstart];
     794           0 :                 qg->list.kids[cnt].qg_cnt = l-lstart;
     795           0 :                 ++cnt;
     796           0 :                 lstart = l;
     797           0 :                 size = qg->qg[l].size;
     798           0 :             } else if ( size!=qg->qg[l].size )
     799           0 :                 size = qg->qg[l].size;
     800             :         }
     801           0 :         if ( size!=-1 ) {
     802           0 :             sprintf( buffer, _("Size: %d (%d)"), size, l-lstart );
     803           0 :             qg->list.kids[cnt].name = copy(buffer);
     804           0 :             qg->list.kids[cnt].parent = &qg->list;
     805           0 :             qg->list.kids[cnt].first = &qg->qg[lstart];
     806           0 :             qg->list.kids[cnt].qg_cnt = l-lstart;
     807             :         }
     808             :     }
     809           0 :     if ( qg->list.kid_cnt==1 )
     810           0 :         qg->list.kids[0].open = true;
     811           0 :     for ( k=0; k<qg->list.kid_cnt; ++k )
     812           0 :         QGSecondLevel(qg,&qg->list.kids[k]);
     813           0 :     QG_Remetric(qg);
     814           0 : }
     815             : 
     816           0 : static int QGSorter(GGadget *g, GEvent *e) {
     817           0 :     if ( e->u.control.subtype == et_listselected ) {
     818           0 :         QGData *qg = GDrawGetUserData(GGadgetGetWindow(g));
     819           0 :         QGDoSort(qg);
     820           0 :         GDrawRequestExpose(qg->v,NULL,false);
     821             :     }
     822           0 : return( true );
     823             : }
     824             : 
     825           0 : static void QGDrawWindow(GWindow pixmap, QGData *qg, GEvent *e) {
     826             :     int l, y, depth;
     827             :     char buffer[200];
     828             :     GRect old, r;
     829             :     struct navigate where;
     830             :     struct qgnode *parent;
     831             : 
     832           0 :     GDrawPushClip(pixmap,&e->u.expose.rect,&old);
     833           0 :     r.width = r.height = qg->as;
     834           0 :     y = qg->as;
     835           0 :     memset(&where,0,sizeof(where));
     836           0 :     QG_FindLine(&qg->list,qg->loff_top,&where);
     837             : 
     838           0 :     for ( l=0; l<qg->vlcnt && where.parent!=NULL; ++l ) {
     839           0 :         for ( parent=where.parent, depth= -2; parent!=NULL; parent=parent->parent, ++depth );
     840           0 :         if ( where.offset==-1 ) {
     841           0 :             r.x = 2+depth*qg->fh;   r.y = y-qg->as+1;
     842           0 :             GDrawDrawRect(pixmap,&r,0x000000);
     843           0 :             GDrawDrawLine(pixmap,r.x+2,r.y+qg->as/2,r.x+qg->as-2,r.y+qg->as/2,
     844             :                     0x000000);
     845           0 :             if ( !where.parent->open )
     846           0 :                 GDrawDrawLine(pixmap,r.x+qg->as/2,r.y+2,r.x+qg->as/2,r.y+qg->as-2,
     847             :                         0x000000);
     848           0 :             GDrawDrawText8(pixmap,r.x+qg->fh,y,where.parent->name,-1, 0x000000);
     849             :         } else {
     850           0 :             QuestionableGrid *q = &where.parent->first[where.offset];
     851           0 :             sprintf( buffer, _("\"%.40s\" size=%d point=%d (%d,%d) distance=%g"),
     852           0 :                     q->sc->name, q->size, q->nearestpt, q->x, q->y, q->distance );
     853           0 :             GDrawDrawText8(pixmap,2+(depth+1)*qg->fh,y,buffer,-1, 0x000000);
     854             :         }
     855           0 :         y += qg->fh;
     856           0 :         QG_NextLine(&where);
     857             :     }
     858           0 :     GDrawPopClip(pixmap,&old);
     859           0 : }
     860             : 
     861           0 : static void QGMouse( QGData *qg, GEvent *e) {
     862           0 :     int l = qg->loff_top + e->u.mouse.y/qg->fh;
     863             :     struct navigate where;
     864             : 
     865           0 :     if ( (e->type == et_mousedown) && (e->u.mouse.button==1)) {
     866           0 :         memset(&where,0,sizeof(where));
     867           0 :         QG_FindLine(&qg->list,l,&where);
     868           0 :         if ( where.parent==NULL )
     869           0 : return;
     870           0 :         if ( where.offset==-1 ) {
     871           0 :             where.parent->open = !where.parent->open;
     872           0 :             QG_Remetric(qg);
     873           0 :             GDrawRequestExpose(qg->v,NULL,false);
     874           0 : return;
     875             :         } else {
     876           0 :             QuestionableGrid *q = &where.parent->first[where.offset];
     877             :             CharView *cv;
     878           0 :             if ( qg->inprocess )
     879           0 : return;
     880           0 :             cv = qg->cv;
     881           0 :             if ( cv==NULL && qg->fv!=NULL ) {
     882           0 :                 qg->inprocess = true;
     883           0 :                 cv = qg->cv = CharViewCreate(q->sc,(FontView *) (qg->fv),qg->fv->map->backmap[q->sc->orig_pos]);
     884           0 :                 if ( qg->layer == ly_fore ) {
     885           0 :                     cv->b.drawmode = dm_fore;
     886             :                 } else {
     887           0 :                     cv->b.layerheads[dm_back] = &qg->sc->layers[qg->layer];
     888           0 :                     cv->b.drawmode = dm_back;
     889             :                 }
     890           0 :                 cv->qg = qg;
     891           0 :                 qg->inprocess = false;
     892           0 :             } else if ( qg->cv==NULL )
     893           0 : return;
     894           0 :             else if ( qg->cv->b.sc != q->sc ) {
     895           0 :                 CVChangeSC(qg->cv,q->sc);
     896             :             }
     897           0 :             cv->ft_pointsizex = cv->ft_pointsizey = q->size;
     898           0 :             cv->ft_ppemy = cv->ft_ppemx = rint(q->size*qg->dpi/72.0);
     899           0 :             cv->ft_dpi = qg->dpi;
     900           0 :             cv->ft_depth = qg->depth;
     901           0 :             cv->note_x = q->x; cv->note_y = q->y;
     902           0 :             cv->show_ft_results = true; cv->showgrids = true;
     903           0 :             CVGridFitChar(cv);
     904             :         }
     905             :     }
     906             : }
     907             : 
     908           0 : static int qgv_e_h(GWindow gw, GEvent *event) {
     909           0 :     QGData *qg = (QGData *) GDrawGetUserData(gw);
     910             : 
     911           0 :     switch ( event->type ) {
     912             :       case et_expose:
     913           0 :         QGDrawWindow(gw,qg,event);
     914           0 :       break;
     915             :       case et_mouseup:
     916             :       case et_mousedown:
     917             :       case et_mousemove:
     918           0 :         QGMouse(qg,event);
     919           0 :       break;
     920             :       case et_char:
     921           0 : return( false );
     922             :       break;
     923             :       case et_resize: {
     924           0 :         int vlcnt = event->u.resize.size.height/qg->fh;
     925           0 :         qg->vlcnt = vlcnt;
     926           0 :         QG_SetSb(qg);
     927           0 :         GDrawRequestExpose(qg->v,NULL,false);
     928           0 :       } break;
     929             :     }
     930           0 : return( true );
     931             : }
     932             : 
     933           0 : static int QG_OK(GGadget *g, GEvent *e) {
     934             : 
     935           0 :     if ( e->type==et_controlevent && e->u.control.subtype == et_buttonactivate ) {
     936           0 :         QGData *qg = GDrawGetUserData(GGadgetGetWindow(g));
     937           0 :         qg->done = true;
     938             :     }
     939           0 : return( true );
     940             : }
     941             : 
     942           0 : static int qg_e_h(GWindow gw, GEvent *event) {
     943             : 
     944           0 :     if ( event->type==et_close ) {
     945           0 :         QGData *qg = (QGData *) GDrawGetUserData(gw);
     946           0 :         qg->done = true;
     947           0 :     } else if ( event->type == et_char ) {
     948           0 : return( false );
     949             :     }
     950           0 : return( true );
     951             : }
     952             : 
     953           0 : static void StartDeltaDisplay(QGData *qg) {
     954             :     GWindowAttrs wattrs;
     955             :     GRect pos;
     956           0 :     GWindow gw, oldgw = qg->gw;
     957             :     GGadgetCreateData gcd[8], boxes[5], *harray[4], *harray2[5], *butarray[8],
     958             :             *varray[4];
     959             :     GTextInfo label[8];
     960             :     int i, k;
     961             :     FontRequest rq;
     962             :     int as, ds, ld;
     963             :     static GFont *valfont=NULL;
     964             :     static int sorts_translated = 0;
     965             : 
     966           0 :     if (!sorts_translated)
     967             :     {
     968           0 :         for (i=0; i<sizeof(sorts)/sizeof(sorts[0]); i++)
     969           0 :             sorts[i].text = (unichar_t *) _((char *) sorts[i].text);
     970           0 :         for (i=0; i<sizeof(glyphsorts)/sizeof(glyphsorts[0]); i++)
     971           0 :             glyphsorts[i].text = (unichar_t *) _((char *) glyphsorts[i].text);
     972           0 :         sorts_translated=1;
     973             :     }
     974             : 
     975           0 :     memset(&wattrs,0,sizeof(wattrs));
     976           0 :     wattrs.mask = wam_events|wam_cursor|wam_utf8_wtitle|wam_undercursor|wam_isdlg;
     977           0 :     wattrs.event_masks = -1;
     978           0 :     wattrs.cursor = ct_mypointer;
     979           0 :     wattrs.utf8_window_title = _("Potential spots for Delta instructions");
     980           0 :     wattrs.is_dlg = true;
     981           0 :     wattrs.undercursor = 1;
     982           0 :     pos.x = pos.y = 0;
     983           0 :     pos.width = GDrawPointsToPixels(NULL,200);
     984           0 :     pos.height = GDrawPointsToPixels(NULL,300);
     985           0 :     qg->gw = gw = GDrawCreateTopWindow(NULL,&pos,qg_e_h,qg,&wattrs);
     986           0 :     qg->done = false;
     987             : 
     988           0 :     if ( valfont==NULL ) {
     989           0 :         memset(&rq,0,sizeof(rq));
     990           0 :         rq.utf8_family_name = "Helvetica";
     991           0 :         rq.point_size = 11;
     992           0 :         rq.weight = 400;
     993           0 :         valfont = GDrawInstanciateFont(gw,&rq);
     994           0 :         valfont = GResourceFindFont("Validate.Font",valfont);
     995             :     }
     996           0 :     qg->font = valfont;
     997           0 :     GDrawWindowFontMetrics(gw,qg->font,&as,&ds,&ld);
     998           0 :     qg->fh = as+ds;
     999           0 :     qg->as = as;
    1000             : 
    1001           0 :     memset(&label,0,sizeof(label));
    1002           0 :     memset(&gcd,0,sizeof(gcd));
    1003           0 :     memset(&boxes,0,sizeof(boxes));
    1004             : 
    1005           0 :     k = 0;
    1006           0 :     label[k].text = (unichar_t *) _("Sort:");
    1007           0 :     label[k].text_is_1byte = true;
    1008           0 :     gcd[k].gd.label = &label[k];
    1009           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
    1010           0 :     gcd[k++].creator = GLabelCreate;
    1011             : 
    1012           0 :     sorts[0].selected = true; sorts[1].selected = sorts[2].selected = false;
    1013           0 :     sorts[2].disabled = qg->fv==NULL;
    1014           0 :     gcd[k].gd.u.list = sorts;
    1015           0 :     gcd[k].gd.cid = CID_Sort;
    1016           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
    1017           0 :     gcd[k].gd.handle_controlevent = QGSorter;
    1018           0 :     gcd[k++].creator = GListButtonCreate;
    1019             : 
    1020           0 :     label[k].text = (unichar_t *) _("Glyph:");
    1021           0 :     label[k].text_is_1byte = true;
    1022           0 :     gcd[k].gd.label = &label[k];
    1023           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
    1024           0 :     gcd[k++].creator = GLabelCreate;
    1025             : 
    1026           0 :     gcd[k].gd.u.list = glyphsorts;
    1027           0 :     gcd[k].gd.flags = gg_enabled|gg_visible;
    1028           0 :     gcd[k].gd.cid = CID_GlyphSort;
    1029           0 :     gcd[k].gd.handle_controlevent = QGSorter;
    1030           0 :     gcd[k++].creator = GListButtonCreate;
    1031           0 :     if ( qg->fv==NULL )
    1032           0 :         gcd[k-1].gd.flags = gcd[k-2].gd.flags = gg_enabled;
    1033           0 :     harray2[0] = &gcd[k-4]; harray2[1] = &gcd[k-3]; harray2[2] = &gcd[k-2]; harray2[3] = &gcd[k-1]; harray2[4] = NULL;
    1034             : 
    1035             : 
    1036           0 :     gcd[k].gd.flags = gg_visible | gg_enabled;
    1037           0 :     gcd[k].gd.u.drawable_e_h = qgv_e_h;
    1038           0 :     gcd[k++].creator = GDrawableCreate;
    1039             : 
    1040           0 :     gcd[k].gd.flags = gg_visible | gg_enabled | gg_sb_vert;
    1041           0 :     gcd[k].gd.handle_controlevent = QG_VScroll;
    1042           0 :     gcd[k++].creator = GScrollBarCreate;
    1043           0 :     harray[0] = &gcd[k-2]; harray[1] = &gcd[k-1]; harray[2] = NULL; harray[3] = NULL;
    1044             : 
    1045           0 :     gcd[k].gd.flags = gg_visible | gg_enabled | gg_but_default;
    1046           0 :     label[k].text = (unichar_t *) _("_OK");
    1047           0 :     label[k].text_is_1byte = true;
    1048           0 :     label[k].text_in_resource = true;
    1049           0 :     gcd[k].gd.label = &label[k];
    1050           0 :     gcd[k].gd.handle_controlevent = QG_OK;
    1051           0 :     gcd[k++].creator = GButtonCreate;
    1052           0 :     butarray[0] = GCD_Glue; butarray[1] = &gcd[k-1]; butarray[2] = GCD_Glue; butarray[3] = NULL;
    1053             : 
    1054           0 :     boxes[2].gd.flags = gg_enabled|gg_visible;
    1055           0 :     boxes[2].gd.u.boxelements = harray;
    1056           0 :     boxes[2].creator = GHVGroupCreate;
    1057             : 
    1058           0 :     boxes[3].gd.flags = gg_enabled|gg_visible;
    1059           0 :     boxes[3].gd.u.boxelements = butarray;
    1060           0 :     boxes[3].creator = GHBoxCreate;
    1061             : 
    1062           0 :     boxes[4].gd.flags = gg_enabled|gg_visible;
    1063           0 :     boxes[4].gd.u.boxelements = harray2;
    1064           0 :     boxes[4].creator = GHBoxCreate;
    1065           0 :     varray[0] = &boxes[4]; varray[1] = &boxes[2]; varray[2] = &boxes[3]; varray[3] = NULL;
    1066             : 
    1067           0 :     boxes[0].gd.flags = gg_enabled|gg_visible;
    1068           0 :     boxes[0].gd.u.boxelements = varray;
    1069           0 :     boxes[0].creator = GVBoxCreate;
    1070             : 
    1071           0 :     GGadgetsCreate(gw,boxes);
    1072           0 :     qg->vsb = gcd[5].ret;
    1073           0 :     qg->v = GDrawableGetWindow(gcd[4].ret);
    1074           0 :     GHVBoxSetExpandableRow(boxes[0].ret,1);
    1075           0 :     GHVBoxSetExpandableCol(boxes[2].ret,0);
    1076           0 :     GHVBoxSetPadding(boxes[2].ret,0,0);
    1077           0 :     GHVBoxSetExpandableCol(boxes[3].ret,gb_expandglue);
    1078           0 :     GHVBoxFitWindow(boxes[0].ret);
    1079             : 
    1080           0 :     QGDoSort(qg);
    1081             : 
    1082           0 :     GDrawSetVisible(gw,true);
    1083           0 :     while ( !qg->done )
    1084           0 :         GDrawProcessOneEvent(NULL);
    1085           0 :     GDrawDestroyWindow(gw);
    1086             : 
    1087           0 :     qgnodeFree(&qg->list);
    1088           0 :     qg->gw = oldgw;
    1089           0 : }

Generated by: LCOV version 1.10