This file is indexed.

/usr/share/gcompris/python/mining.py is in gcompris-data 15.02-1.1ubuntu1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

   1
   2
   3
   4
   5
   6
   7
   8
   9
  10
  11
  12
  13
  14
  15
  16
  17
  18
  19
  20
  21
  22
  23
  24
  25
  26
  27
  28
  29
  30
  31
  32
  33
  34
  35
  36
  37
  38
  39
  40
  41
  42
  43
  44
  45
  46
  47
  48
  49
  50
  51
  52
  53
  54
  55
  56
  57
  58
  59
  60
  61
  62
  63
  64
  65
  66
  67
  68
  69
  70
  71
  72
  73
  74
  75
  76
  77
  78
  79
  80
  81
  82
  83
  84
  85
  86
  87
  88
  89
  90
  91
  92
  93
  94
  95
  96
  97
  98
  99
 100
 101
 102
 103
 104
 105
 106
 107
 108
 109
 110
 111
 112
 113
 114
 115
 116
 117
 118
 119
 120
 121
 122
 123
 124
 125
 126
 127
 128
 129
 130
 131
 132
 133
 134
 135
 136
 137
 138
 139
 140
 141
 142
 143
 144
 145
 146
 147
 148
 149
 150
 151
 152
 153
 154
 155
 156
 157
 158
 159
 160
 161
 162
 163
 164
 165
 166
 167
 168
 169
 170
 171
 172
 173
 174
 175
 176
 177
 178
 179
 180
 181
 182
 183
 184
 185
 186
 187
 188
 189
 190
 191
 192
 193
 194
 195
 196
 197
 198
 199
 200
 201
 202
 203
 204
 205
 206
 207
 208
 209
 210
 211
 212
 213
 214
 215
 216
 217
 218
 219
 220
 221
 222
 223
 224
 225
 226
 227
 228
 229
 230
 231
 232
 233
 234
 235
 236
 237
 238
 239
 240
 241
 242
 243
 244
 245
 246
 247
 248
 249
 250
 251
 252
 253
 254
 255
 256
 257
 258
 259
 260
 261
 262
 263
 264
 265
 266
 267
 268
 269
 270
 271
 272
 273
 274
 275
 276
 277
 278
 279
 280
 281
 282
 283
 284
 285
 286
 287
 288
 289
 290
 291
 292
 293
 294
 295
 296
 297
 298
 299
 300
 301
 302
 303
 304
 305
 306
 307
 308
 309
 310
 311
 312
 313
 314
 315
 316
 317
 318
 319
 320
 321
 322
 323
 324
 325
 326
 327
 328
 329
 330
 331
 332
 333
 334
 335
 336
 337
 338
 339
 340
 341
 342
 343
 344
 345
 346
 347
 348
 349
 350
 351
 352
 353
 354
 355
 356
 357
 358
 359
 360
 361
 362
 363
 364
 365
 366
 367
 368
 369
 370
 371
 372
 373
 374
 375
 376
 377
 378
 379
 380
 381
 382
 383
 384
 385
 386
 387
 388
 389
 390
 391
 392
 393
 394
 395
 396
 397
 398
 399
 400
 401
 402
 403
 404
 405
 406
 407
 408
 409
 410
 411
 412
 413
 414
 415
 416
 417
 418
 419
 420
 421
 422
 423
 424
 425
 426
 427
 428
 429
 430
 431
 432
 433
 434
 435
 436
 437
 438
 439
 440
 441
 442
 443
 444
 445
 446
 447
 448
 449
 450
 451
 452
 453
 454
 455
 456
 457
 458
 459
 460
 461
 462
 463
 464
 465
 466
 467
 468
 469
 470
 471
 472
 473
 474
 475
 476
 477
 478
 479
 480
 481
 482
 483
 484
 485
 486
 487
 488
 489
 490
 491
 492
 493
 494
 495
 496
 497
 498
 499
 500
 501
 502
 503
 504
 505
 506
 507
 508
 509
 510
 511
 512
 513
 514
 515
 516
 517
 518
 519
 520
 521
 522
 523
 524
 525
 526
 527
 528
 529
 530
 531
 532
 533
 534
 535
 536
 537
 538
 539
 540
 541
 542
 543
 544
 545
 546
 547
 548
 549
 550
 551
 552
 553
 554
 555
 556
 557
 558
 559
 560
 561
 562
 563
 564
 565
 566
 567
 568
 569
 570
 571
 572
 573
 574
 575
 576
 577
 578
 579
 580
 581
 582
 583
 584
 585
 586
 587
 588
 589
 590
 591
 592
 593
 594
 595
 596
 597
 598
 599
 600
 601
 602
 603
 604
 605
 606
 607
 608
 609
 610
 611
 612
 613
 614
 615
 616
 617
 618
 619
 620
 621
 622
 623
 624
 625
 626
 627
 628
 629
 630
 631
 632
 633
 634
 635
 636
 637
 638
 639
 640
 641
 642
 643
 644
 645
 646
 647
 648
 649
 650
 651
 652
 653
 654
 655
 656
 657
 658
 659
 660
 661
 662
 663
 664
 665
 666
 667
 668
 669
 670
 671
 672
 673
 674
 675
 676
 677
 678
 679
 680
 681
 682
 683
 684
 685
 686
 687
 688
 689
 690
 691
 692
 693
 694
 695
 696
 697
 698
 699
 700
 701
 702
 703
 704
 705
 706
 707
 708
 709
 710
 711
 712
 713
 714
 715
 716
 717
 718
 719
 720
 721
 722
 723
 724
 725
 726
 727
 728
 729
 730
 731
 732
 733
 734
 735
 736
 737
 738
 739
 740
 741
 742
 743
 744
 745
 746
 747
 748
 749
 750
 751
 752
 753
 754
 755
 756
 757
 758
 759
 760
 761
 762
 763
 764
 765
 766
 767
 768
 769
 770
 771
 772
 773
 774
 775
 776
 777
 778
 779
 780
 781
 782
 783
 784
 785
 786
 787
 788
 789
 790
 791
 792
 793
 794
 795
 796
 797
 798
 799
 800
 801
 802
 803
 804
 805
 806
 807
 808
 809
 810
 811
 812
 813
 814
 815
 816
 817
 818
 819
 820
 821
 822
 823
 824
 825
 826
 827
 828
 829
 830
 831
 832
 833
 834
 835
 836
 837
 838
 839
 840
 841
 842
 843
 844
 845
 846
 847
 848
 849
 850
 851
 852
 853
 854
 855
 856
 857
 858
 859
 860
 861
 862
 863
 864
 865
 866
 867
 868
 869
 870
 871
 872
 873
 874
 875
 876
 877
 878
 879
 880
 881
 882
 883
 884
 885
 886
 887
 888
 889
 890
 891
 892
 893
 894
 895
 896
 897
 898
 899
 900
 901
 902
 903
 904
 905
 906
 907
 908
 909
 910
 911
 912
 913
 914
 915
 916
 917
 918
 919
 920
 921
 922
 923
 924
 925
 926
 927
 928
 929
 930
 931
 932
 933
 934
 935
 936
 937
 938
 939
 940
 941
 942
 943
 944
 945
 946
 947
 948
 949
 950
 951
 952
 953
 954
 955
 956
 957
 958
 959
 960
 961
 962
 963
 964
 965
 966
 967
 968
 969
 970
 971
 972
 973
 974
 975
 976
 977
 978
 979
 980
 981
 982
 983
 984
 985
 986
 987
 988
 989
 990
 991
 992
 993
 994
 995
 996
 997
 998
 999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
#  gcompris - mining.py
#
# Copyright (C) 2012 Peter Albrecht
#   based on template by Bruno Coudoin
#
#   This program is free software; you can redistribute it and/or modify
#   it under the terms of the GNU General Public License as published by
#   the Free Software Foundation; either version 3 of the License, or
#   (at your option) any later version.
#
#   This program is distributed in the hope that it will be useful,
#   but WITHOUT ANY WARRANTY; without even the implied warranty of
#   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
#   GNU General Public License for more details.
#
#   You should have received a copy of the GNU General Public License
#   along with this program; if not, see <http://www.gnu.org/licenses/>.
#
# mining activity.

import math
import gobject
import gtk
import gtk.gdk
import gcompris
import gcompris.bonus
import gcompris.utils
import gcompris.skin
import gcompris.sound
import goocanvas
import pango
import random
import cairo
import mining_tutorial

from mining_tools import Area, BlockingArea

from gcompris import gcompris_gettext as _

class Gcompris_mining:
  """ GCompis Mining-Activity """

  # The factor to shrink the source image with, in order to make it fit on the screen.
  # This has to be larger than 1 (= source image has higher resolution than screen),
  # so the image looks still nice, if we zoom in a bit.
  source_image_scale = 3.0

  # the distance, the mouse cursor has to approach the nugget, triggering the next tutorial step
  # (in 800x520 coordinate space) (should be in sync with the graphics in tutorial.svgz)
  min_nugget_approach = 50.0


  def __init__(self, gcomprisBoard):
    """ Constructor """

    # Save the gcomprisBoard, it defines everything we need
    # to know from the core
    self.gcomprisBoard = gcomprisBoard

    # Needed to get key_press
    gcomprisBoard.disable_im_context = True


    ##
    # initialize and document instance variables

    self.nuggets_to_collect = 0
    """ the number of nuggets, we need to collect in this level """

    self.nugget_count = 0
    """ the number of nuggets, we already have collected """

    self.need_new_nugget = False
    """ used to trigger the creation of a new nugget at next opportunity (only at max zoomed out) """

    self.is_game_won = False
    """ used to start new game, after game was won and bonus is displayed """

    self.__is_game_paused = False
    """ used to avoid input (like scrolling) during game pause """

    self.tutorial = None
    """ handle to the tutorial object """

    self.is_tutorial_startable = False
    """ can the tutorial be started in this level? """

    self.is_tutorial_enabled = False
    """ has the tutorial been started for this nugget? """

    self.last_mouse_pos_x = None
    """ the x position of the mouse pointer, the last time, we saw it (800x520) """

    self.last_mouse_pos_y = None
    """ the y position of the mouse pointer, the last time, we saw it (800x520) """


  def start(self):
    """ Load data and start the activity """

    self.gcomprisBoard.maxlevel = 3
    self.gcomprisBoard.sublevel = 1
    self.gcomprisBoard.number_of_sublevel = 1

    # Set the buttons we want in the bar.
    # We need to set BAR_LEVEL, in order to see the "help" icon!?!
    gcompris.bar_set(gcompris.BAR_LEVEL)

    gcompris.bar_location(0, -1, 0.8)

    # Setup a nugget-blocking-area for the GCompris bar, to avoid placing the nugget behind
    # the bar.
    # The bar is in the lower, left corner of the screen with dimensions: width = 196; height = 50
    self.gc_bar_blocker = BlockingArea(0, 470, 196, 520)

    # Create our rootitem. We put each canvas item in it so at the end we
    # only have to kill it. The canvas deletes all the items it contains
    # automatically.
    self.rootitem = goocanvas.Group(parent = self.gcomprisBoard.canvas.get_root_item())
    self.rootitem.connect("button_press_event", self.on_button_press)
    self.rootitem.connect("motion_notify_event", self.on_mouse_move)

    svghandle = gcompris.utils.load_svg("mining/rockwall.svgz")

    self.viewport = Viewport(self, self.rootitem)

    rockwall_img = goocanvas.Svg(
      parent = self.viewport.get_gc_group(),
      svg_handle = svghandle,
      svg_id = "#BACKGROUND"
      )

    # the further zoom logic highly depends on the following conditions
    assert(gcompris.BOARD_WIDTH == (rockwall_img.get_bounds().x2 / self.source_image_scale))
    assert(gcompris.BOARD_HEIGHT == (rockwall_img.get_bounds().y2 / self.source_image_scale))

    self.lorry = Lorry(svghandle, self.rootitem)

    self.placer = Placer(self)

    self.decorations = Decorations(svghandle, self.viewport.get_gc_group(), self.placer)

    self.nugget = Nugget(svghandle, self.viewport.get_gc_group())

    # create sparkling last, so it is on above the nugget
    self.sparkling = Sparkling(svghandle, self.viewport.get_gc_group())

    # prepare the tutorial
    self.tutorial = mining_tutorial.MiningTutorial(self.rootitem)

    # prepare the teacher icon to start the tutorial with
    self.teacher_img = goocanvas.Image(
      parent = self.rootitem,
      pixbuf = gcompris.utils.load_pixmap("mining/tux-teacher.png"),
      x = 850,
      y = 850,
      visibility = goocanvas.ITEM_INVISIBLE
      )
    self.teacher_img.scale(0.5, 0.5)
    self.teacher_img.connect("button_press_event", self.start_tutorial)
    gcompris.utils.item_focus_init(self.teacher_img, None)

    # initialize the level, to start with
    self.set_level(1)


  def set_level(self, level):
    """ Initialize new level and start it. """

    self.end_level()
    self.is_tutorial_enabled = False

    # store new level
    self.gcomprisBoard.level = level
    gcompris.bar_set_level(self.gcomprisBoard);

    # load level specific values
    if level == 1:
      self.nuggets_to_collect = 3
      self.is_tutorial_startable = True

      # add the tutorials blocking area, to avoid the nugget being placed behind
      # the tutorial mouse or touchpad
      self.placer.add_blocker(self.tutorial.get_blocking_area())

    elif level == 2:
      self.nuggets_to_collect = 6
      self.is_tutorial_startable = False

    elif level == 3:
      self.nuggets_to_collect = 9
      self.is_tutorial_startable = False

    else:
      print("Warning: No level specific values defined for level %i! Keeping current settings." % level)


    # prepare new game
    self.is_game_won = False
    self.nugget_count = 0
    self.need_new_nugget = False
    self.nugget.hide()

    self.update_lorry()
    self.viewport.reset(level)

    self.placer.add_blocker(self.gc_bar_blocker)
    self.placer.add_blocker(self.lorry)
    for blocking_area in self.viewport.get_nugget_blocker():
      self.placer.add_blocker(blocking_area)

    self.decorations.decorate_viewport(10)

    self.place_new_nugget()


  def place_new_nugget(self):
    """ Place a new nugget to collect on the rockwall """

    self.placer.place(self.nugget, self.nugget.reset)

    (x, y) = self.nugget.get_sparkling_coordinates()
    self.sparkling.reset(x, y)
    self.sparkling.animation_start()
    self.need_new_nugget = False

    # The following sound was copied form "Tuxpaint" (GPL)
    gcompris.sound.play_ogg("mining/realrainbow.ogg")

    # tutorial stuff
    self.is_tutorial_enabled = False
    if self.is_tutorial_startable:
      self.show_teacher_icon()
    else:
      self.hide_teacher_icon()


  def start_tutorial(self, item, target_item, event):
    """
    Start the tutorial (teacher icon has been clicked)
      item - The element connected with this callback function
      target_item - The element under the cursor
      event  - gtk.gdk.Event
    """

    self.hide_teacher_icon()
    self.is_tutorial_enabled = True
    self.tutorial.start()
    nuggetArea = Area(self.nugget.get_bounds())

    # determine state, to start the tutorial from
    if self.need_new_nugget:
      # we have collected the nugget, and need to zoom out
      self.tutorial.set_tutorial_state('zoom out', True)

    elif self.nugget.is_visible():
      # the nugget needs to be collected
      self.tutorial.set_tutorial_state('click', True)
    else:
      # we are at the beginning: 'move to' and then 'zoom in'

      # remember them for potential "move to" tutorial animation restarts
      self.last_mouse_pos_x = event.x_root
      self.last_mouse_pos_y = event.y_root

      self.tutorial.set_tutorial_state('move to', True, event.x_root, event.y_root, nuggetArea.center_x, nuggetArea.center_y)

    # we processed this click event
    return True


  def show_teacher_icon(self):
    """ Show the teacher icon (the icon to start the tutorial with """
    self.teacher_img.props.visibility = goocanvas.ITEM_VISIBLE


  def hide_teacher_icon(self):
    """ Hide the teacher icon (the icon to start the tutorial with """
    self.teacher_img.props.visibility = goocanvas.ITEM_INVISIBLE


  def on_mouse_move(self, item, target_item, event):
    """
    The user moved the mouse
      item - The element connected with this callback function
      target_item - The element under the cursor
      event  - gtk.gdk.Event
    """

    if not self.is_tutorial_enabled:
      return True


    ##
    # if we the mouse cursor is close enough to the nugget, switch to next tutorial step

    # event.x / .y are based on the target_item, which may be the rockwall or a stone, ...
    # so we can't use those, since the stones have another coordinate space  :(

    # get the coordinates relative to the root of the screen (800 x 520)
    x = event.x_root
    y = event.y_root

    # remember them for potential "move to" tutorial animation restarts
    self.last_mouse_pos_x = x
    self.last_mouse_pos_y = y

    # get_bounds() also gives us coordinates relative to the root of the screen (800 x 520)
    nuggetArea = Area(self.nugget.get_bounds())
    nx = nuggetArea.center_x
    ny = nuggetArea.center_y

    #          a^2         +         b^2         <=                         c^2
    if (x - nx) * (x - nx) + (y - ny) * (y - ny) <= self.min_nugget_approach * self.min_nugget_approach:
      # the mouse cursor is close enough, go to next tutorial step

      if self.nugget.is_visible():
        # the nugget is already visible, so we can fast-forward to the "zoom in" tutorial
        self.tutorial.set_tutorial_state('zoom in', False)
        self.tutorial.set_tutorial_state('click', False)
      else:
        # go the default way: "move to", "zoom in", "click", ...
        self.tutorial.set_tutorial_state('zoom in', False)

    else:
      # if we still want to show the user, where to move the mouse pointer to, we need to
      # update this animation now
      if self.tutorial.get_tutorial_state() == 'move to':
        self.restart_tutorial_move_to(x, y)

    # we processed this event
    return True


  def restart_tutorial_move_to(self, mouse_x, mouse_y):
    """
    Start a timer to restart the 'move to' tutorial animation
      mouse_x, mouse_y: position of the mouse pointer relative to the root of the screen (800 x 520)
    """
    # Since event messages like "scroll_event" seem to have to be processed completely, until *.get_bounds()
    # delivers us updated coordinates, we have to call the animation restart in another event, following
    # the current one. So we set up a timer, waiting 1 millisecond, doing the job. ;)
    gobject.timeout_add(1, self.__restart_tutorial_move_to_on_timer, mouse_x, mouse_y)


  def __restart_tutorial_move_to_on_timer(self, mouse_x, mouse_y):
    """
    Restart the 'move to' tutorial animation (called by a timeout-event from restart_tutorial_move_to())
      mouse_x, mouse_y: position of the mouse pointer relative to the root of the screen (800 x 520)
    """
    nuggetArea = Area(self.nugget.get_bounds())
    nx = nuggetArea.center_x
    ny = nuggetArea.center_y
    self.tutorial.restart_tutorial_step(mouse_x, mouse_y, nx, ny)

    # do not call this timer again
    return False


  def on_zoom_change(self, state):
    """ Do something according to specific zoom states (E.g. the nugget is only visible at maximum zoom.) """
    # As of 2012-08-11 there seems to be no "gcomrpis way" to change the mouse cursor to
    # a individual png. So we can't change to a pickaxe. :(

    if self.is_tutorial_enabled:
      if self.tutorial.get_tutorial_state() == 'move to':
        self.restart_tutorial_move_to(self.last_mouse_pos_x, self.last_mouse_pos_y)


    if state == 'min':
      self.nugget.hide()

      if self.need_new_nugget:
        # nugget has been collected
        if self.is_tutorial_enabled:
          self.tutorial.stop()

        self.place_new_nugget()

    elif state == 'mid':
      self.nugget.hide()

    elif state == 'max':
      if self.is_tutorial_enabled:
        # try to proceed to next tutorial step (from "zoom in")
        self.tutorial.set_tutorial_state('click', False)

      self.nugget.show()

    else:
      assert(False)


  def on_button_press(self, item, target_item, event = None):
    """
    The user clicked somewhere.

    Since the spark sometimes hides the nugget, click events might be misleaded to the spark-image.
    So we can't use "nugget.connect()". :(
    """

    # ignore input while game paused
    if self.is_game_paused():
      return True

    # don't react on double clicks
    if event.type != gtk.gdk.BUTTON_PRESS:
      return False

    # coordinate space fix
    # Event-coordinates are in the coordinate space of that item, which lies direct under the cursor.
    # To "normalize" them, we translate those coordinates to canvas coordinate space.
    (x, y) = self.gcomprisBoard.canvas.convert_from_item_space(target_item, event.x, event.y)

    if self.nugget.is_visible():
      # Was the mouse-click over the nugget?
      # An alternative would be:
      #    gcomprisBoard.canvas.get_items_at(x, y, True)
      # But we don't have the nugget_img object here.
      if self.nugget.is_hit(x, y):
        self.collect_nugget()

    # we processed this event
    return True


  def collect_nugget(self):
    """ The nugget was clicked, so collect it """

    self.placer.remove_blocker(self.nugget)
    self.nugget_count += 1
    self.sparkling.animation_stop()
    self.nugget.hide()

    self.update_lorry()

    if self.nugget_count >= self.nuggets_to_collect:
      self.on_level_won()
    else:
      # The following sound is based on "metalpaint.wav" from "Tuxpaint" (GPL)
      gcompris.sound.play_ogg("mining/pickaxe.ogg")

      # we need to collect more nuggets, so lets place a new one
      self.need_new_nugget = True

      if self.is_tutorial_enabled:
        self.tutorial.set_tutorial_state('zoom out', False)


  def update_lorry(self):
    """ Updates the nugget-collect-counter of the lorry in the lower right corner """
    self.lorry.update_text(str(self.nugget_count) + "/" + str(self.nuggets_to_collect))


  def on_level_won(self):
    """ The user collected enough nuggets """

    if self.is_tutorial_enabled:
      self.tutorial.stop()

    self.is_game_won = True;

    # In GCompris version up to and including 12.5, there is a bug with the LION bonus:
    # The bonus-constant "LION" does not exist, but had to be called by "GNU"
    try:
      # try to display bonus with correct constant
      gcompris.bonus.display(gcompris.bonus.WIN, gcompris.bonus.LION)
    except AttributeError:
      # fail back to GCompris version <= 12.5
      gcompris.bonus.display(gcompris.bonus.WIN, gcompris.bonus.GNU)


  def pause(self, pause):
    """ Called by GCompris, when the game is paused (e.g. config or bonus display) """

    # When the bonus is displayed, this function is called first with pause(1) and then with pause(0)
    if pause == 0:
      # pause finished
      self.__is_game_paused = False

      if self.is_game_won:
        # the bonus has been shown, so start a new game in the next level
        self.set_level(self.get_next_level())

    else:
      # pause started
      self.__is_game_paused = True


  def is_game_paused(self):
    """ Determine if the game is currently paused """
    return self.__is_game_paused


  def get_next_level(self):
    """ Determines the next level """

    next_level = self.gcomprisBoard.level + 1
    if(next_level > self.gcomprisBoard.maxlevel):
      next_level = self.gcomprisBoard.maxlevel

    return next_level


  def end_level(self):
    """ Terminate the current level """
    self.tutorial.stop()
    self.sparkling.end()
    self.placer.remove_all_blocker()
    self.decorations.cleanup_viewport()


  def end(self):
    """ Terminate the activity """

    self.end_level()

    self.tutorial.end()

    # Remove the root item removes all the others inside it
    self.rootitem.remove()


  # -----------------------
  # GCompris interface functions, not needed by mining-activity

  def config_start(self, profile):
    pass

  def config_stop(self):
    pass

  def key_press(self, keyval, commit_str, preedit_str):
    pass

  def ok(self):
    pass

  def repeat(self):
    pass



class Lorry:
  """ The lorry in the lower right corner of the screen, showing the number of collected nuggets """

  # position of the lorry (in screen coordinates (800/520))
  x = 730.0
  y = 470.0

  # center of the lorry in the svg file (used for rotation and positioning)
  pivot_x = 2220
  pivot_y = 1432

  # the svg file is scaled with a factor of 3 (to enable nice zoom in) so we
  # have to correct this for the lorry, which is not affected by zooming.
  # Plus: the lorry designed in the svg file is a little bit to small for the
  # text in default font to fit in.
  scale = 0.4


  def __init__(self, svghandle, rootitem):
    """
    Constructor:
      svghandle          : handle of the svg file, containing graphics data
      rootitem           : the root item to attach goo-object to
    """

    self.lorry_img = goocanvas.Svg(
      parent = rootitem,
      svg_handle = svghandle,
      svg_id = "#LORRY",
      )

    self.text = goocanvas.Text(
      parent = rootitem,
      font = gcompris.skin.get_font("gcompris/board/medium"),
      x = self.x + 9,
      y = self.y + 2,
      anchor = gtk.ANCHOR_CENTER,
      fill_color = "white",
      text = "-/-"
      )

    self.__update_transformation()


  def get_bounds(self):
    """ Get the bounds of the lorry image on the canvas """
    return self.lorry_img.get_bounds()


  def update_text(self, text):
    """ Set a new text for the lorry to display """
    self.text.set_properties(text = text)


  def __update_transformation(self):
    """ Updates the transformation matrix of the lorry """

    m_center_to_origin = cairo.Matrix(1, 0, 0, 1, -self.pivot_x, -self.pivot_y)
    m_scale = cairo.Matrix(self.scale, 0, 0, self.scale, 0, 0)
    m_to_destination = cairo.Matrix(1, 0, 0, 1, self.x, self.y)

    # combine all transformation matrices to one
    matrix = m_center_to_origin * m_scale * m_to_destination

    self.lorry_img.set_transform(matrix)



class Placer:
  """ This class randomly places items on the screen and assures, that they do not overlap """

  def __init__(self, activity):
    """
    Constructor:
      activity : the main activity class
    """
    self.activity = activity


    # initialize and document instance variables

    self.blocking_areas = []
    """ the internal list of blocking areas """


  def place(self, item, place_callback):
    """
    Place "item" on the screen.
      item            : the item to place (needs to implement "get_bounds()")
      place_callback  : The callback function to actually place the item (and modify, like rotate/scale).
                        Is called with parameters (item, x, y).
    """
    area = Area(item.get_bounds())
    width_half = int(area.width / 2.0)
    height_half = int(area.height / 2.0)

    safety_counter = 0

    while True:
      # get new random position for the item
      x = random.randrange(width_half, gcompris.BOARD_WIDTH - width_half) * self.activity.source_image_scale
      y = random.randrange(height_half, gcompris.BOARD_HEIGHT - height_half) * self.activity.source_image_scale

      # place the item to this position
      place_callback(item, x, y)

      # check for overlapping objects
      if not self.__does_object_collide_with_registered_ones(item):
        # we found a valid position without collisions
        # lets remember the positioned item...
        self.add_blocker(item);
        # ... and end the search for a valid position
        break

      safety_counter += 1
      if safety_counter > 20:
        # We tried to place this object many times, but could not find a valid position.
        # Seems to be very difficult with this amount of objects.
        # Since an invalid position (= overlapping objects) is way better than a frozen
        # application, we exit this while loop with an "invalid" position.
        print("Warning: safety_counter reached maximum!")
        self.add_blocker(item);
        break


  def add_blocker(self, blocking_area):
    """
    Add a new blocking area to the internal list of blocking areas
      blocking_area: Object that implement method get_bounds() (like goocanvas.Item.get_bounds())
    """
    self.blocking_areas.append(blocking_area)


  def remove_blocker(self, blocking_area):
    """ Removes the given blocking area from the internal list of blocking areas """
    for i in range(len(self.blocking_areas)):
      if self.blocking_areas[i] == blocking_area:
        # this is the blocking_area, we are looking for, so remove it from the list
        self.blocking_areas.pop(i)
        return

    print("Warning: blocking-area not in list: " + str(blocking_area))


  def remove_all_blocker(self):
    """ Removes all blocker from the internal list """
    self.blocking_areas = []


  def __does_object_collide_with_registered_ones(self, asked_object):
    """ Tests whether any registered object collides with the asked_object """
    area_a = Area(asked_object.get_bounds())

    for blocking_area in self.blocking_areas:
      area_r = Area(blocking_area.get_bounds())

      # collision on x-axis?
      if math.fabs(area_a.center_x - area_r.center_x) <= (area_r.width + area_a.width) / 2.0:
        # collision on y-axis?
        if math.fabs(area_a.center_y - area_r.center_y) <= (area_r.height + area_a.height) / 2.0:
          # collision!
          return True

    # no collision
    return False



class Viewport:
  """ The viewport handles zooming in and out with the appropriate translation """

  def __init__(self, activity, parent):
    """
    Constructor:
      activity           : the main activity object
      parent             : the parent GooCanvas item to add our gc_group
    """
    # initialize and document instance variables

    self.source_image_scale = activity.source_image_scale
    """ see documentation in Gcompris_mining """

    self.x = 0
    """ viewport x translation """

    self.y = 0
    """ viewport y translation """

    self.scale = 1.0
    """ current viewport scale / zoom """

    self.scale_min = 1.0 / self.source_image_scale
    """ The limit to max zoom out, while still filling all the screen with the rockwall. """

    self.scale_max = None
    """
    The limit to max zoom in.
    Try to keep scale_max reachable by  scale_min * zoom_factor ^ n  (with n in [1, 2, 3, 4, ...[)
    This value is set up in reset()
    """

    self.zoom_factor = None
    """ The factor to zoom on each zoom event. This value is set up in reset() """


    self.gc_group = goocanvas.Group(parent = parent)
    """ The GooCanvas group, which holds everything that is affected by zooming """


    self.gcomprisBoard = activity.gcomprisBoard
    self.cb_zoom_change = activity.on_zoom_change
    self.is_game_paused = activity.is_game_paused

    self.gc_group.connect("scroll_event", self.__on_scroll)

    self.nugget_blocker = (
      BlockingArea(0, 0, 800, 42), # top
      BlockingArea(758, 0, 800, 520), # right
      BlockingArea(0, 510, 800, 520), # bottom
      BlockingArea(0, 0, 42, 520)     # left
    )


  def reset(self, level):
    """ Reset the viewport and set zooming parameters according to current level """

    # in general: higher level means more zooming
    if level == 1:
      self.scale_max = 0.6000
      self.zoom_factor = 1.3417  # 2 zoom events from min to max

    elif level == 2:
      self.scale_max = 0.8000
      self.zoom_factor = 1.2447  # 4 zoom events from min to max

    elif level == 3:
      self.scale_max = 1.0000
      self.zoom_factor = 1.2010  # 6 zoom events from min to max

    else:
      print("Warning: No level specific zooming values defined for level %i! Keeping current settings." % level)


    # zoom out max
    self.scale = self.scale_min
    self.x = 0
    self.y = 0

    self.__update_transformation()


  def get_gc_group(self):
    """ get the GooCanvas group, which holds everything that is affected by zooming """
    return self.gc_group


  def get_nugget_blocker(self):
    """ Get the viewport's blocking areas, where no nugget should be placed """
    return self.nugget_blocker


  def __on_scroll(self, item, target_item, event = None):
    """
    Called at scroll (= zoom) events
      item - The element connected with this callback function
      target_item - The element under the cursor
      event  - gtk.gdk.Event
    """

    # ignore input while game paused
    if self.is_game_paused():
      return True

    assert(event.type == gtk.gdk.SCROLL)


    ###
    # coordinate space fix

    # Event-coordinates are in the coordinate space of that item, which lies direct under the cursor.
    # So if the user scrolls over the spark, for example, we have to translate those coordinates to
    # the coordinate space of our gc_group.
    (x, y) = self.gcomprisBoard.canvas.convert_from_item_space(target_item, event.x, event.y)
    (x, y) = self.gcomprisBoard.canvas.convert_to_item_space(self.gc_group, x, y)


    ###
    # remember old values for calculation

    old_scale = self.scale
    old_x = self.x
    old_y = self.y


    ###
    # calculate the new scale factor

    if event.direction == gtk.gdk.SCROLL_DOWN:
      # zoom out
      self.scale /= self.zoom_factor
      if self.scale < self.scale_min:
        self.scale = self.scale_min
    elif event.direction == gtk.gdk.SCROLL_UP:
      # zoom in
      self.scale *= self.zoom_factor
      if self.scale > self.scale_max:
        self.scale = self.scale_max
    else:
      # We don't accept SRCOLL_RIGHT and LEFT
      return True


    ###
    # The pixel under the cursor shall be fixed, while everything else zooms in/out
    # (we zoom at the mouse position), so we have to calculate a new position for
    # gc_group (displaying primarily the rockwall image).

    new_x = old_x + x * (old_scale - self.scale)
    new_y = old_y + y * (old_scale - self.scale)


    ###
    # Checks to prevent empty areas on any side of the rockwall

    min_x = gcompris.BOARD_WIDTH * (1 - self.scale * self.source_image_scale)
    min_y = gcompris.BOARD_HEIGHT * (1 - self.scale * self.source_image_scale)

    # block uncovered area on the left
    if new_x > 0:
      new_x = 0

    # block uncovered area on the top
    if new_y > 0:
      new_y = 0

    # block uncovered area on the right
    if new_x < min_x:
      new_x = min_x

    # block uncovered area on the bottom
    if new_y < min_y:
      new_y = min_y


    ###
    # apply new values

    self.x = new_x
    self.y = new_y

    self.__update_transformation()


    ###
    # inform main class about zoom change

    if old_scale != self.scale:
      if self.scale == self.scale_max:
        state = 'max'
      elif self.scale == self.scale_min:
        state = 'min'
      else:
        state = 'mid'

      # call the callback function of main class
      self.cb_zoom_change(state)

    return True


  def __update_transformation(self):
    """ Update the viewport's transformation matrix """

    self.gc_group.set_simple_transform(self.x, self.y, self.scale, 0)



class Decorations:
  """
  This class handles decorations, like stones. They have the meaning of:
    - make every level look a bit different
    - help orienting, while scrolling
  """

  # our decoration types in the svg file
  decoration_types = (
    {
      'svg_id': '#STONE1',
      'svg_x': 500,
      'svg_y': 1300
    },
    {
      'svg_id': '#STONE2',
      'svg_x': 1000,
      'svg_y': 1300
    },
    {
      'svg_id': '#STONE3',
      'svg_x': 1500,
      'svg_y': 1300
    },
    {
      'svg_id': '#STONE4',
      'svg_x': 2000,
      'svg_y': 1300
    },
  )


  def __init__(self, svghandle, gc_group, placer):
    """
    Constructor:
      - svghandle : handle to the svg file, to load the decoration pictures
      - gc_group  : our viewport's gc_group, the decorations are attached to
      - placer    : reference to the Placer object
    """
    self.svghandle = svghandle
    self.placer = placer


    ##
    # initialize and document instance variables

    self.viewport_gc_group = gc_group
    """ The viewport's GooCanvas Group to add our decoration_group to """

    self.decoration_group = None
    """ A goocanvas group, that holds all our decoration, so we can easily remove them, by removing only this group. """

    self.current_decoration_id = None
    """ ID of the decoration type, currently being placed. (Used to overcome callback bounderies) """


  def decorate_viewport(self, number_of_decorations):
    """ Fill the viewport with some decorations """

    assert(self.decoration_group == None)
    self.decoration_group = goocanvas.Group(parent = self.viewport_gc_group)

    for i in range(number_of_decorations):
      # select random decoration
      self.current_decoration_id = random.randrange(4)
      svg_id = self.decoration_types[self.current_decoration_id]['svg_id']

      # create decoration
      decoration = goocanvas.Svg(
        parent = self.decoration_group,
        svg_handle = self.svghandle,
        svg_id = svg_id,
        )

      self.placer.place(decoration, self.__place_decoration)


  def __place_decoration(self, decoration, x, y):
    """ Updates the transformation of the decoration to the new coordinates. Rotation and scale are varied. """

    svg_x = self.decoration_types[self.current_decoration_id]['svg_x']
    svg_y = self.decoration_types[self.current_decoration_id]['svg_y']


    # scale between 0.5 and 2.0
    scale = math.pow(1.2, random.randrange(-4, +4))

    rot = random.randrange(-20, +20)


    # we need those values more than once, so lets remember them
    a = math.radians(rot)
    cos_a = math.cos(a)
    sin_a = math.sin(a)

    # create the transformation matrices
    m_center_to_origin = cairo.Matrix(1, 0, 0, 1, -svg_x, -svg_y)
    m_scale = cairo.Matrix(scale, 0, 0, scale, 0, 0)
    m_rotate = cairo.Matrix(cos_a, sin_a, -sin_a, cos_a, 0, 0)
    m_to_destination = cairo.Matrix(1, 0, 0, 1, x, y)

    # combine all transformation matrices to one
    matrix = m_center_to_origin * m_scale * m_rotate * m_to_destination

    decoration.set_transform(matrix)


  def cleanup_viewport(self):
    """ Remove all decorations from the viewport """
    if self.decoration_group != None:
      self.decoration_group.remove()
      self.decoration_group = None



class Nugget:
  """ The gold nugget """

  # center of the spark in the svg file (used for rotation and positioning)
  pivot_x = 1000
  pivot_y = 800


  def __init__(self, svghandle, parent):
    """
    Constructor:
      svghandle      : handle to the svg file, holding the pictures
      parent         : GooCanvas parent item of the gold nugget
    """

    # initialize and document instance variables

    self.x = 0.0
    """ x position of the nugget (in the rockwall/gc_group) """

    self.y = 0.0
    """ y position of the nugget (in the rockwall/gc_group) """

    self.nugget_img = goocanvas.Svg(
      parent = parent,
      svg_handle = svghandle,
      svg_id = "#NUGGET",

      # start invisible, since x/y are not set properly yet
      visibility = goocanvas.ITEM_INVISIBLE
      )
    """ picture of the nugget """


  def reset(self, nugget, x, y):
    """
    Move nugget to a new position.
      nugget : we don't need it in this case, since there is only one nugget image in this class,
                but the callback interface defines this parameter
      x      : new x position for the nugget image
      y      : new y position for the nugget image
    """
    self.x = x
    self.y = y
    self.nugget_img.set_simple_transform(self.x - self.pivot_x, self.y - self.pivot_y, 1.0, 0)


  def get_bounds(self):
    """ Get the bounds of the nugget image on the canvas """
    return self.nugget_img.get_bounds()


  def show(self):
    """ Display the nugget """
    self.nugget_img.props.visibility = goocanvas.ITEM_VISIBLE


  def hide(self):
    """ Hide the nugget """
    self.nugget_img.props.visibility = goocanvas.ITEM_INVISIBLE


  def is_visible(self):
    """ Tells whether the nugget is visible (True) or hidden (False) """
    return self.nugget_img.props.visibility == goocanvas.ITEM_VISIBLE


  def is_hit(self, x, y):
    """
    Determines whether the given coordinates are in the nugget's bounding box.
      x and y are the coordinates in the canvas coordinate space.
    """

    # get the nugget's bounding box in canvas coordinate space
    bounds = self.nugget_img.get_bounds()

    if bounds.x1 <= x and x <= bounds.x2 and bounds.y1 <= y and y <= bounds.y2:
      return True

    return False


  def get_sparkling_coordinates(self):
    """ Get the coordinates, where the sparkling should appear """
    return (self.x - 10, self.y - 10)


class Sparkling:
  """
  Handles all "sparkling"-stuff of the gold nugget.

  The sparkling animation consists of "sparkphases", the time when the spark can be seen,
  and pauses, the time when the spark is hidden.
  """

  # value to initialize the rotation delta with
  rot_delta_init = 6

  # the factor, the spark shrinks every animation step
  scale_factor = 0.90

  # minimum scale factor; if reached, the current spark-phase ends
  scale_min = 0.4

  # the scale factor to start a spark-phase with
  scale_max = 1.0

  # the time (in milliseconds) between two sparkling animation steps
  timer_milliseconds = 30

  # the number of timer-events, a pause lasts
  pause_ticks_total = 25

  # add some randomness to pause_ticks_total
  pause_tick_variation = 10

  # center of the spark in the svg file (used for rotation and positioning)
  pivot_x = 600
  pivot_y = 600


  def __init__(self, svghandle, parent):
    """
    Constructor:
      svghandle : handle to the svg file, holding the pictures
      parent    : GooCanvas parent item of this spark
    """

    # initialize and document instance variables

    self.spark = goocanvas.Svg(
      parent = parent,
      svg_handle = svghandle,
      svg_id = "#SPARK",

      # start invisible, since x/y are not set properly yet
      visibility = goocanvas.ITEM_INVISIBLE
      )
    """ the spark image in the GooCanvas """

    self.x = 0.0
    """ x position of the spark (in the rockwall/gc_group) """

    self.y = 0.0
    """ y position of the spark (in the rockwall/gc_group) """

    self.angle = 0
    """ rotation (in degrees) """

    self.rot_delta = 0
    """ the amount to rotate at every animation step """

    self.scale = 0
    """ the sparks current scale factor """

    self.timer = None
    """ the timer object, firing timeout-events for our animation """

    self.pause_ticks_current = 0
    """ counts the number of elapsed pause ticks between two spark-phases """


  def end(self):
    """ Our "destructor" """
    if self.__is_animation_playing():
      self.animation_stop()


  def reset(self, x, y):
    """ Prepare a new sparkling animation. """

    self.x = x
    self.y = y

    self.pause_ticks_current = 0

    # we turn rotation direction at every sparkphase
    self.rot_delta = self.rot_delta_init

    self.__sparkphase_start()


  def __update_transformation(self):
    """ Updates the transformation matrix of the spark (= calculate new picture of animation). """

    # we need those values more than once, so lets remember them
    a = math.radians(self.angle)
    cos_a = math.cos(a)
    sin_a = math.sin(a)

    # create the transformation matrices
    m_center_to_origin = cairo.Matrix(1, 0, 0, 1, -self.pivot_x, -self.pivot_y)
    m_scale = cairo.Matrix(self.scale, 0, 0, self.scale, 0, 0)
    m_rotate = cairo.Matrix(cos_a, sin_a, -sin_a, cos_a, 0, 0)
    m_to_destination = cairo.Matrix(1, 0, 0, 1, self.x, self.y)

    # combine all transformation matrices to one
    matrix = m_center_to_origin * m_scale * m_rotate * m_to_destination

    self.spark.set_transform(matrix)


  def __show_spark(self):
    """ Display the spark """
    self.spark.props.visibility = goocanvas.ITEM_VISIBLE


  def __hide_spark(self):
    """ Hide the spark """
    self.spark.props.visibility = goocanvas.ITEM_INVISIBLE


  def __sparkphase_start(self):
    """ Start a new sparkphase, showing the spark """

    # setup new values for this sparkphase
    self.angle += random.randrange(0, 359)
    self.rot_delta *= -1
    self.scale = self.scale_max

    # show the spark at the new state
    self.__update_transformation()
    self.__show_spark()


  def __calculate_new_state(self):
    """ Calculate next animation step """
    self.scale *= self.scale_factor
    self.angle += self.rot_delta


  def __animate(self):
    """ Called every x milliseconds, to animate the sparkling """

    if self.pause_ticks_current > 0:
      # pause this tick
      self.pause_ticks_current -= 1

      if self.pause_ticks_current <= 0:
        # pause ends
        self.__sparkphase_start()

      return True

    # no pause

    if self.scale < self.scale_min:
      # start pause
      self.pause_ticks_current = self.pause_ticks_total + random.randrange(-self.pause_tick_variation, self.pause_tick_variation)
      self.__hide_spark()
      return True

    # normal sparkle animation
    self.__calculate_new_state()
    self.__update_transformation()

    # call timeout again
    return True


  def animation_start(self):
    """ Starts the sparkling animation """
    assert(not self.__is_animation_playing())

    self.timer = gobject.timeout_add(self.timer_milliseconds, self.__animate)


  def animation_stop(self):
    """ Stops the sparkling animation """
    assert(self.__is_animation_playing())

    gobject.source_remove(self.timer)
    self.timer = None

    self.__hide_spark()


  def __is_animation_playing(self):
    """ Tells us, if there is an animation running at the moment """
    return self.timer != None