aboutsummaryrefslogtreecommitdiff
path: root/src/player-item.vala
blob: 83b19d6246169a96728f01d71a7b03e443194daa (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
158
159
160
/*
Copyright 2010 Canonical Ltd.

Authors:
    Conor Curran <conor.curran@canonical.com>

This program is free software: you can redistribute it and/or modify it 
under the terms of the GNU General Public License version 3, as published 
by the Free Software Foundation.

This program is distributed in the hope that it will be useful, but 
WITHOUT ANY WARRANTY; without even the implied warranties of 
MERCHANTABILITY, SATISFACTORY QUALITY, 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/>.
*/

using Dbusmenu;
using Gee;
using Gdk;

public class PlayerItem : Dbusmenu.Menuitem
{
	public PlayerController owner {get; construct;}
	public string item_type { get; construct; }
	private const int EMPTY = -1;
	private FetchFile fetcher;
	private string previous_temp_album_art_path;

	public PlayerItem(string type)
	{		
		Object(item_type: type);
	}
	
	construct {
		this.property_set(MENUITEM_PROP_TYPE, item_type);
		this.previous_temp_album_art_path = null;
	}

	public void reset(HashSet<string> attrs){		
		foreach(string s in attrs){
			debug("attempting to set prop %s to EMPTY", s);
			this.property_set_int(s, EMPTY);
		}
	}
	
	/**
	 * update()
	 * Base update method for playeritems, takes the attributes and the incoming updates
	 * and attmepts to update the appropriate props on the object. 
	 * Album art is handled separately to deal with remote and local file paths.
	 */
	public void update(HashTable<string, Value?> data, HashSet<string> attributes)
	{
		debug("PlayerItem::update()");
		if(data == null){
			debug("PlayerItem::Update -> The hashtable was null - just leave it!");
			return;
		}		
		
		foreach(string property in attributes){
			string[] input_keys = property.split("-");
			string search_key = input_keys[input_keys.length-1 : input_keys.length][0];
			debug("search key = %s", search_key);
			Value? v = data.lookup(search_key);
			
			if (v.holds (typeof (string))){
				string update = v.get_string().strip();
				debug("with value : %s", update);
				if(property.contains("mpris:artUrl")){
					  this.fetch_art(update.strip(), property);
					 	continue;					                     
				}
				this.property_set(property, update);											
			}			    
			else if (v.holds (typeof (int))){
				debug("with value : %i", v.get_int());
				this.property_set_int(property, v.get_int());
			}
			else if (v.holds (typeof (int64))){
				debug("with value : %i", (int)v.get_int64());
				this.property_set_int(property, (int)v.get_int64());
			}
			else if(v.holds (typeof (bool))){
				debug("with value : %s", v.get_boolean().to_string());				
				this.property_set_bool(property, v.get_boolean());
			}
		}
		if(this.property_get_bool(MENUITEM_PROP_VISIBLE) == false){
			this.property_set_bool(MENUITEM_PROP_VISIBLE, true);
		}
	}	
	
	public bool populated(HashSet<string> attrs)
	{
		foreach(string prop in attrs){
			debug("populated ? - prop: %s", prop);
		  int value_int = property_get_int(prop);
			debug("populated ? - prop %s and value %i", prop, value_int);
			if(property_get_int(prop) != EMPTY){
				return true;
			}
		}
		return false;
	}

	public void fetch_art(string uri, string prop)
	{		
		File art_file = File.new_for_uri(uri);
		if(art_file.is_native() == true){
			string path;		
			try{
				path = Filename.from_uri(uri.strip());			
				this.property_set(prop, path);			
			}
			catch(ConvertError e){
				warning("Problem converting URI %s to file path",
					      uri); 
			}
			// eitherway return, the artwork was local
			return;			
		}
		// otherwise go remote
		this.fetcher = new FetchFile (uri, prop);
		this.fetcher.failed.connect (() => { this.on_fetcher_failed ();});
		this.fetcher.completed.connect (this.on_fetcher_completed);
		this.fetcher.fetch_data ();		
	}
	
	private void on_fetcher_failed ()
	{
		warning("on_fetcher_failed -> could not fetch artwork");	
	}

	private void on_fetcher_completed(ByteArray update, string property)
	{
		try{
			PixbufLoader loader = new PixbufLoader ();
			loader.write (update.data, update.len);
			loader.close ();
			Pixbuf icon = loader.get_pixbuf ();				
 			string path = Environment.get_user_special_dir(UserDirectory.PICTURES).dup().concat("/indicator-sound-XXXXXX");
			int r = FileUtils.mkstemp(path);		
			icon.save (path, loader.get_format().get_name());		
			if(this.previous_temp_album_art_path != null){
				FileUtils.remove(this.previous_temp_album_art_path);
			}			
			this.previous_temp_album_art_path = path;
			this.property_set(property, path);
		}
	  catch(GLib.Error e){
			warning("Problem creating file from bytearray fetched from the interweb - error: %s",
			        e.message);
		}				
	}		
	
}