aboutsummaryrefslogtreecommitdiff
path: root/demos/photoAlpha.tcl
blob: df3f04b5365542a726c8fd7621342c4a016ffbcb (plain)
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
# $Id: photoAlpha.tcl 1826 2006-10-18 11:58:23Z lecoanet $
# this simple demo has been developped by P.Lecoanet <lecoanet@cena.fr>

if {![info exists zincDemo]} {
    error "This script should be run from the zinc-widget demo."
}

namespace eval photoAlpha {
    package require Img

    variable girl [image create photo -file [file join $::zinc_demos images photoAlpha.png]]
    variable texture [image create photo -file [file join $::zinc_demos images stripped_texture.gif]]

    variable w .photoAlpha
    catch {destroy $w}
    toplevel $w
    wm title $w "Zinc photo transparency Demonstration"
    wm iconname $w photoAlpha

    variable defaultfont [font create -family Helvetica -size 16 -weight normal]

    grid [button $w.dismiss -text Dismiss -command "destroy $w"] -row 2 -column 0 -pady 10
    grid [button $w.code -text "See Code" -command "showCode $w"] -row 2 -column 1 -pady 10


    ###########################################
    # Text zone
    #######################
    ####################

    grid [text $w.text -relief sunken -borderwidth 2 -height 7] \
	-row 0 -column 0 -columnspan 2 -sticky ew

    $w.text insert end {This demo needs openGL for displaying the photo
        with transparent pixels and for rescaling/rotating.
        You can transform this png photo with your mouse:
        Drag-Button 1 for moving the photo,
        Drag-Button 2 for zooming the photo,
        Drag-Button 3 for rotating the photo,
        Shift-Drag-Button 1 for modifying the global photo transparency.}


    ###########################################
    # Zinc
    ##########################################
    zinc $w.zinc -width 350 -height 250 -render 1 -font $defaultfont \
	    -borderwidth 3 -relief sunken -tile $texture
    grid $w.zinc -row 1 -column 0 -columnspan 2 -sticky news
    grid columnconfigure $w 0 -weight 1
    grid columnconfigure $w 1 -weight 1
    grid rowconfigure $w 1 -weight 2

    variable topGroup [$w.zinc add group 1]

    variable girlItem [$w.zinc add icon $topGroup -image $girl \
			   -composescale 1 -composerotation 1]

    #
    # Controls for the window transform.
    #
    bind $w.zinc <ButtonPress-1>  "::photoAlpha::press motion %x %y"
    bind $w.zinc <ButtonRelease-1>  ::photoAlpha::release
    bind $w.zinc <ButtonPress-2>  "::photoAlpha::press zoom %x %y"
    bind $w.zinc <ButtonRelease-2>  ::photoAlpha::release
    bind $w.zinc <ButtonPress-3>  "::photoAlpha::press mouseRotate %x %y"
    bind $w.zinc <ButtonRelease-3>  ::photoAlpha::release

    #
    # Controls for alpha and gradient
    #
    bind $w.zinc <Shift-ButtonPress-1> "::photoAlpha::press modifyAlpha %x %y"
    bind $w.zinc <Shift-ButtonRelease-1> ::photoAlpha::release


    variable curX 0
    variable curY 0
    variable curAngle 0

    proc press {action x y} {
	variable w
	variable curAngle
	variable curX
	variable curY

	set curX $x
	set curY $y
	set curAngle [expr atan2($y, $x)]
	bind $w.zinc <Motion> "::photoAlpha::$action %x %y"
    }

    proc motion {x y} {
	variable w
	variable topGroup
	variable curX
	variable curY

	foreach {x1 y1 x2 y2} [$w.zinc transform $topGroup \
				   [list $x $y $curX $curY]] break
	$w.zinc translate $topGroup [expr $x1 - $x2] [expr $y1 - $y2]
	set curX $x
	set curY $y
    }

    proc zoom {x y} {
	variable w
	variable topGroup
	variable curX
	variable curY

	if {$x > $curX} {
	    set maxX $x
	} else {
	    set maxX $curX
	}
	if {$y > $curY} {
	    set maxY $y
	} else {
	    set maxY $curY
	}
	if {($maxX == 0) || ($maxY == 0)} {
	    return;
	}
	set sx [expr 1.0 + (double($x - $curX) / $maxX)]
	set sy [expr 1.0 + (double($y - $curY) / $maxY)]
	$w.zinc scale $topGroup $sx $sx

	set curX $x
	set curY $y
    }

    proc mouseRotate {x y} {
	variable w
	variable curAngle
	variable topGroup

	set lAngle [expr atan2($y, $x)]
	$w.zinc rotate $topGroup [expr $lAngle - $curAngle]
	set curAngle  $lAngle
    }

    proc release {} {
	variable w

	bind $w.zinc <Motion> {}
    }

    proc modifyAlpha {x y} {
	variable w
	variable topGroup

	set xRate [expr double($x) / [$w.zinc cget -width]]
	set xRate [expr ($xRate < 0) ? 0 : ($xRate > 1) ? 1 : $xRate]
	set alpha [expr int($xRate * 100)]

	$w.zinc itemconfigure $topGroup -alpha $alpha
    }
}