| Index: base/mime_util_linux.cc
|
| ===================================================================
|
| --- base/mime_util_linux.cc (revision 0)
|
| +++ base/mime_util_linux.cc (revision 0)
|
| @@ -0,0 +1,564 @@
|
| +// Copyright (c) 2009 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "base/mime_util.h"
|
| +
|
| +#include <sys/time.h>
|
| +#include <time.h>
|
| +
|
| +#include <cstdlib>
|
| +#include <list>
|
| +#include <map>
|
| +#include <vector>
|
| +
|
| +#include "base/file_util.h"
|
| +#include "base/logging.h"
|
| +#include "base/scoped_ptr.h"
|
| +#include "base/singleton.h"
|
| +#include "base/string_util.h"
|
| +#include "base/third_party/xdg_mime/xdgmime.h"
|
| +
|
| +namespace {
|
| +
|
| +class IconTheme;
|
| +
|
| +class MimeUtilConstants {
|
| + public:
|
| +
|
| + // In seconds, specified by icon theme specs.
|
| + const int kUpdateInterval;
|
| +
|
| + // Store icon directories and their mtimes.
|
| + std::map<FilePath, int>* icon_dirs;
|
| +
|
| + // Store icon formats.
|
| + std::vector<std::string>* icon_formats;
|
| +
|
| + // Store loaded icon_theme.
|
| + std::map<std::string, IconTheme*>* icon_themes;
|
| +
|
| + static const size_t kDefaultThemeNum = 5;
|
| +
|
| + // The default theme.
|
| + IconTheme* default_themes[kDefaultThemeNum];
|
| +
|
| + time_t last_check_time;
|
| +
|
| + private:
|
| + MimeUtilConstants()
|
| + : kUpdateInterval(5),
|
| + icon_dirs(NULL),
|
| + icon_formats(NULL),
|
| + icon_themes(NULL),
|
| + last_check_time(0) {
|
| + }
|
| + friend struct DefaultSingletonTraits<MimeUtilConstants>;
|
| + DISALLOW_COPY_AND_ASSIGN(MimeUtilConstants);
|
| +};
|
| +
|
| +// IconTheme represents an icon theme as defined by the xdg icon theme spec.
|
| +// Example themes on GNOME include 'Human' and 'Mist'.
|
| +// Example themes on KDE include 'crystalsvg' and 'kdeclassic'.
|
| +class IconTheme {
|
| + public:
|
| + // A theme consists of multiple sub-directories, like '32x32' and 'scalable'.
|
| + class SubDirInfo {
|
| + public:
|
| + // See spec for details.
|
| + enum Type {
|
| + Fixed,
|
| + Scalable,
|
| + Threshold
|
| + };
|
| + SubDirInfo()
|
| + : size(0),
|
| + type(Threshold),
|
| + max_size(0),
|
| + min_size(0),
|
| + threshold(2) {
|
| + }
|
| + size_t size; // Nominal size of the icons in this directory.
|
| + Type type; // Type of the icon size.
|
| + size_t max_size; // Maximum size that the icons can be scaled to.
|
| + size_t min_size; // Minimum size that the icons can be scaled to.
|
| + size_t threshold; // Maximum difference from desired size. 2 by default.
|
| + };
|
| +
|
| + explicit IconTheme(const std::string& name);
|
| +
|
| + ~IconTheme() {
|
| + delete[] info_array_;
|
| + }
|
| +
|
| + // Returns the path to an icon with the name |icon_name| and a size of |size|
|
| + // pixels. If the icon does not exist, but |inherits| is true, then look for
|
| + // the icon in the parent theme.
|
| + FilePath GetIconPath(const std::string& icon_name, int size, bool inherits);
|
| +
|
| + // Load a theme with the name |theme_name| into memory. Returns null if theme
|
| + // is invalid.
|
| + static IconTheme* LoadTheme(const std::string& theme_name);
|
| +
|
| + private:
|
| + // Returns the path to an icon with the name |icon_name| in |subdir|.
|
| + FilePath GetIconPathUnderSubdir(const std::string& icon_name,
|
| + const std::string& subdir);
|
| +
|
| + // Whether the theme loaded properly.
|
| + bool IsValid() {
|
| + return index_theme_loaded_;
|
| + }
|
| +
|
| + // Read and parse |file| which is usually named 'index.theme' per theme spec.
|
| + bool LoadIndexTheme(const FilePath& file);
|
| +
|
| + // Checks to see if the icons in |info| matches |size| (in pixels). Returns
|
| + // 0 if they match, or the size difference in pixels.
|
| + size_t MatchesSize(SubDirInfo* info, size_t size);
|
| +
|
| + // Yet another function to read a line.
|
| + std::string ReadLine(FILE* fp);
|
| +
|
| + // Set directories to search for icons to the comma-separated list |dirs|.
|
| + bool SetDirectories(const std::string& dirs);
|
| +
|
| + bool index_theme_loaded_; // True if an instance is properly loaded.
|
| + // store the scattered directories of this theme.
|
| + std::list<FilePath> dirs_;
|
| +
|
| + // store the subdirs of this theme and array index of |info_array_|.
|
| + std::map<std::string, int> subdirs_;
|
| + SubDirInfo* info_array_; // List of sub-directories.
|
| + std::string inherits_; // Name of the theme this one inherits from.
|
| +};
|
| +
|
| +IconTheme::IconTheme(const std::string& name)
|
| + : index_theme_loaded_(false),
|
| + info_array_(NULL) {
|
| + // Iterate on all icon directories to find directories of the specified
|
| + // theme and load the first encountered index.theme.
|
| + std::map<FilePath, int>::iterator iter;
|
| + FilePath theme_path;
|
| + std::map<FilePath, int>* icon_dirs =
|
| + Singleton<MimeUtilConstants>::get()->icon_dirs;
|
| + for (iter = icon_dirs->begin(); iter != icon_dirs->end(); ++iter) {
|
| + theme_path = iter->first.Append(name);
|
| + if (!file_util::DirectoryExists(theme_path))
|
| + continue;
|
| + FilePath theme_index = theme_path.Append("index.theme");
|
| + if (!index_theme_loaded_ && file_util::PathExists(theme_index)) {
|
| + if (!LoadIndexTheme(theme_index))
|
| + return;
|
| + index_theme_loaded_ = true;
|
| + }
|
| + dirs_.push_back(theme_path);
|
| + }
|
| +}
|
| +
|
| +FilePath IconTheme::GetIconPath(const std::string& icon_name, int size,
|
| + bool inherits) {
|
| + std::map<std::string, int>::iterator subdir_iter;
|
| + FilePath icon_path;
|
| +
|
| + for (subdir_iter = subdirs_.begin();
|
| + subdir_iter != subdirs_.end();
|
| + ++subdir_iter) {
|
| + SubDirInfo* info = &info_array_[subdir_iter->second];
|
| + if (MatchesSize(info, size) == 0) {
|
| + icon_path = GetIconPathUnderSubdir(icon_name, subdir_iter->first);
|
| + if (!icon_path.empty())
|
| + return icon_path;
|
| + }
|
| + }
|
| + // Now looking for the mostly matched.
|
| + int min_delta_seen = 9999;
|
| +
|
| + for (subdir_iter = subdirs_.begin();
|
| + subdir_iter != subdirs_.end();
|
| + ++subdir_iter) {
|
| + SubDirInfo* info = &info_array_[subdir_iter->second];
|
| + int delta = abs(MatchesSize(info, size));
|
| + if (delta < min_delta_seen) {
|
| + FilePath path = GetIconPathUnderSubdir(icon_name, subdir_iter->first);
|
| + if (!path.empty()) {
|
| + min_delta_seen = delta;
|
| + icon_path = path;
|
| + }
|
| + }
|
| + }
|
| +
|
| + if (!icon_path.empty() || !inherits || inherits_ == "")
|
| + return icon_path;
|
| +
|
| + IconTheme* theme = LoadTheme(inherits_);
|
| + if (theme)
|
| + return theme->GetIconPath(icon_name, size, inherits);
|
| + else
|
| + return FilePath();
|
| +}
|
| +
|
| +IconTheme* IconTheme::LoadTheme(const std::string& theme_name) {
|
| + scoped_ptr<IconTheme> theme;
|
| + std::map<std::string, IconTheme*>* icon_themes =
|
| + Singleton<MimeUtilConstants>::get()->icon_themes;
|
| + if (icon_themes->find(theme_name) != icon_themes->end()) {
|
| + theme.reset((*icon_themes)[theme_name]);
|
| + } else {
|
| + theme.reset(new IconTheme(theme_name));
|
| + if (!theme->IsValid())
|
| + theme.reset();
|
| + (*icon_themes)[theme_name] = theme.get();
|
| + }
|
| + return theme.release();
|
| +}
|
| +
|
| +FilePath IconTheme::GetIconPathUnderSubdir(const std::string& icon_name,
|
| + const std::string& subdir) {
|
| + FilePath icon_path;
|
| + std::list<FilePath>::iterator dir_iter;
|
| + std::vector<std::string>* icon_formats =
|
| + Singleton<MimeUtilConstants>::get()->icon_formats;
|
| + for (dir_iter = dirs_.begin(); dir_iter != dirs_.end(); ++dir_iter) {
|
| + for (size_t i = 0; i < icon_formats->size(); ++i) {
|
| + icon_path = dir_iter->Append(subdir);
|
| + icon_path = icon_path.Append(icon_name + (*icon_formats)[i]);
|
| + if (file_util::PathExists(icon_path))
|
| + return icon_path;
|
| + }
|
| + }
|
| + return FilePath();
|
| +}
|
| +
|
| +bool IconTheme::LoadIndexTheme(const FilePath& file) {
|
| + FILE* fp = file_util::OpenFile(file, "r");
|
| + SubDirInfo* current_info = NULL;
|
| + if (!fp)
|
| + return false;
|
| +
|
| + // Read entries.
|
| + while (!feof(fp) && !ferror(fp)) {
|
| + std::string buf = ReadLine(fp);
|
| + if (buf == "")
|
| + break;
|
| +
|
| + std::string entry;
|
| + TrimWhitespaceASCII(buf, TRIM_ALL, &entry);
|
| + if (entry.length() == 0 || entry[0] == '#') {
|
| + // Blank line or Comment.
|
| + continue;
|
| + } else if (entry[0] == '[' && info_array_) {
|
| + current_info = NULL;
|
| + std::string subdir = entry.substr(1, entry.length() - 2);
|
| + if (subdirs_.find(subdir) != subdirs_.end())
|
| + current_info = &info_array_[subdirs_[subdir]];
|
| + }
|
| +
|
| + std::string key, value;
|
| + std::vector<std::string> r;
|
| + SplitStringDontTrim(entry, '=', &r);
|
| + if (r.size() < 2)
|
| + continue;
|
| +
|
| + TrimWhitespaceASCII(r[0], TRIM_ALL, &key);
|
| + for (size_t i = 1; i < r.size(); i++)
|
| + value.append(r[i]);
|
| + TrimWhitespaceASCII(value, TRIM_ALL, &value);
|
| +
|
| + if (current_info) {
|
| + if (key == "Size") {
|
| + current_info->size = atoi(value.c_str());
|
| + } else if (key == "Type") {
|
| + if (value == "Fixed")
|
| + current_info->type = SubDirInfo::Fixed;
|
| + else if (value == "Scalable")
|
| + current_info->type = SubDirInfo::Scalable;
|
| + else if (value == "Threshold")
|
| + current_info->type = SubDirInfo::Threshold;
|
| + } else if (key == "MaxSize") {
|
| + current_info->max_size = atoi(value.c_str());
|
| + } else if (key == "MinSize") {
|
| + current_info->min_size = atoi(value.c_str());
|
| + } else if (key == "Threshold") {
|
| + current_info->threshold = atoi(value.c_str());
|
| + }
|
| + } else {
|
| + if (key.compare("Directories") == 0 && !info_array_) {
|
| + if (!SetDirectories(value)) break;
|
| + } else if (key.compare("Inherits") == 0) {
|
| + if (value != "hicolor")
|
| + inherits_ = value;
|
| + }
|
| + }
|
| + }
|
| +
|
| + file_util::CloseFile(fp);
|
| + return info_array_ != NULL;
|
| +}
|
| +
|
| +size_t IconTheme::MatchesSize(SubDirInfo* info, size_t size) {
|
| + if (info->type == SubDirInfo::Fixed) {
|
| + return size - info->size;
|
| + } else if (info->type == SubDirInfo::Scalable) {
|
| + if (size >= info->min_size && size <= info->max_size) {
|
| + return 0;
|
| + } else {
|
| + return abs(size - info->min_size) < abs(size - info->max_size) ?
|
| + (size - info->min_size) : (size - info->max_size);
|
| + }
|
| + } else {
|
| + if (size >= info->size - info->threshold &&
|
| + size <= info->size + info->threshold) {
|
| + return 0;
|
| + } else {
|
| + return abs(size - info->size - info->threshold) <
|
| + abs(size - info->size + info->threshold)
|
| + ? size - info->size - info->threshold
|
| + : size - info->size + info->threshold;
|
| + }
|
| + }
|
| +}
|
| +
|
| +std::string IconTheme::ReadLine(FILE* fp) {
|
| + if (!fp)
|
| + return "";
|
| +
|
| + std::string result = "";
|
| + const size_t kBufferSize = 100;
|
| + char buffer[kBufferSize];
|
| + while ((fgets(buffer, kBufferSize - 1, fp)) != NULL) {
|
| + result += buffer;
|
| + size_t len = result.length();
|
| + if (len == 0)
|
| + break;
|
| + char end = result[len - 1];
|
| + if (end == '\n' || end == '\0')
|
| + break;
|
| + }
|
| +
|
| + return result;
|
| +}
|
| +
|
| +bool IconTheme::SetDirectories(const std::string& dirs) {
|
| + int num = 0;
|
| + std::string::size_type pos = 0, epos;
|
| + std::string dir;
|
| + while ((epos = dirs.find(',', pos)) != std::string::npos) {
|
| + TrimWhitespaceASCII(dirs.substr(pos, epos - pos), TRIM_ALL, &dir);
|
| + if (dir.length() == 0) {
|
| + LOG(WARNING) << "Invalid index.theme: blank subdir";
|
| + return false;
|
| + }
|
| + subdirs_[dir] = num++;
|
| + pos = epos + 1;
|
| + }
|
| + TrimWhitespaceASCII(dirs.substr(pos), TRIM_ALL, &dir);
|
| + if (dir.length() == 0) {
|
| + LOG(WARNING) << "Invalid index.theme: blank subdir";
|
| + return false;
|
| + }
|
| + subdirs_[dir] = num++;
|
| + info_array_ = new SubDirInfo[num];
|
| + return true;
|
| +}
|
| +
|
| +// Make sure |dir| exists and add it to the list of icon directories.
|
| +void TryAddIconDir(const FilePath& dir) {
|
| + if (!file_util::DirectoryExists(dir))
|
| + return;
|
| + (*Singleton<MimeUtilConstants>::get()->icon_dirs)[dir] = 0;
|
| +}
|
| +
|
| +// For a xdg directory |dir|, add the appropriate icon sub-directories.
|
| +void AddXDGDataDir(const FilePath& dir) {
|
| + if (!file_util::DirectoryExists(dir))
|
| + return;
|
| + TryAddIconDir(dir.Append("icons"));
|
| + TryAddIconDir(dir.Append("pixmaps"));
|
| +}
|
| +
|
| +// Enable or disable SVG support.
|
| +void EnableSvgIcon(bool enable) {
|
| + std::vector<std::string>* icon_formats =
|
| + Singleton<MimeUtilConstants>::get()->icon_formats;
|
| + icon_formats->clear();
|
| + icon_formats->push_back(".png");
|
| + if (enable) {
|
| + icon_formats->push_back(".svg");
|
| + icon_formats->push_back(".svgz");
|
| + }
|
| + icon_formats->push_back(".xpm");
|
| +}
|
| +
|
| +// Add all the xdg icon directories.
|
| +void InitIconDir() {
|
| + Singleton<MimeUtilConstants>::get()->icon_dirs->clear();
|
| + std::string xdg_data_dirs;
|
| + char* env = getenv("XDG_DATA_HOME");
|
| + if (!env) {
|
| + env = getenv("HOME");
|
| + if (env) {
|
| + FilePath local_data_dir(env);
|
| + local_data_dir = local_data_dir.AppendASCII(".local");
|
| + local_data_dir = local_data_dir.AppendASCII("share");
|
| + AddXDGDataDir(local_data_dir);
|
| + }
|
| + } else {
|
| + AddXDGDataDir(FilePath(env));
|
| + }
|
| +
|
| + env = getenv("XDG_DATA_DIRS");
|
| + if (!env) {
|
| + AddXDGDataDir(FilePath("/usr/local/share"));
|
| + AddXDGDataDir(FilePath("/usr/share"));
|
| + } else {
|
| + std::string xdg_data_dirs = env;
|
| + std::string::size_type pos = 0, epos;
|
| + while ((epos = xdg_data_dirs.find(':', pos)) != std::string::npos) {
|
| + AddXDGDataDir(FilePath(xdg_data_dirs.substr(pos, epos - pos)));
|
| + pos = epos + 1;
|
| + }
|
| + AddXDGDataDir(FilePath(xdg_data_dirs.substr(pos)));
|
| + }
|
| +}
|
| +
|
| +// Per xdg theme spec, we should check the icon directories every so often for
|
| +// newly added icons. This isn't quite right.
|
| +void EnsureUpdated() {
|
| + struct timeval t;
|
| + gettimeofday(&t, NULL);
|
| + time_t now = t.tv_sec;
|
| + MimeUtilConstants* constants = Singleton<MimeUtilConstants>::get();
|
| + time_t last_check_time = constants->last_check_time;
|
| +
|
| + if (last_check_time == 0) {
|
| + constants->icon_dirs = new std::map<FilePath, int>;
|
| + constants->icon_themes = new std::map<std::string, IconTheme*>;
|
| + constants->icon_formats = new std::vector<std::string>;
|
| + EnableSvgIcon(true);
|
| + InitIconDir();
|
| + last_check_time = now;
|
| + } else {
|
| + // TODO(thestig): something changed. start over. Upstream fix to Google
|
| + // Gadgets for Linux.
|
| + if (now > last_check_time + constants->kUpdateInterval) {
|
| + }
|
| + }
|
| +}
|
| +
|
| +// Find a fallback icon if we cannot find it in the default theme.
|
| +FilePath LookupFallbackIcon(const std::string& icon_name) {
|
| + FilePath icon;
|
| + MimeUtilConstants* constants = Singleton<MimeUtilConstants>::get();
|
| + std::map<FilePath, int>::iterator iter;
|
| + std::map<FilePath, int>* icon_dirs = constants->icon_dirs;
|
| + std::vector<std::string>* icon_formats = constants->icon_formats;
|
| + for (iter = icon_dirs->begin(); iter != icon_dirs->end(); ++iter) {
|
| + for (size_t i = 0; i < icon_formats->size(); ++i) {
|
| + icon = iter->first.Append(icon_name + (*icon_formats)[i]);
|
| + if (file_util::PathExists(icon))
|
| + return icon;
|
| + }
|
| + }
|
| + return FilePath();
|
| +}
|
| +
|
| +// Initialize the list of default themes.
|
| +void InitDefaultThemes() {
|
| + IconTheme** default_themes =
|
| + Singleton<MimeUtilConstants>::get()->default_themes;
|
| + // TODO(thestig): There is no standard way to know about the current icon
|
| + // theme. So just make a guess. We may be able to do this better. If so,
|
| + // upstream fix to Google Gadgets for Linux.
|
| + char* env = getenv("GGL_ICON_THEME");
|
| + if (env)
|
| + default_themes[0] = IconTheme::LoadTheme(env);
|
| +
|
| + env = getenv("KDE_FULL_SESSION");
|
| + if (env) {
|
| + env = getenv("KDE_SESSION_VERSION");
|
| + if (!env || env[0] != '4') {
|
| + default_themes[1] = IconTheme::LoadTheme("crystalsvg"); // KDE3
|
| + default_themes[2] = IconTheme::LoadTheme("oxygen"); // KDE4
|
| + } else {
|
| + default_themes[1] = IconTheme::LoadTheme("oxygen"); // KDE4
|
| + default_themes[2] = IconTheme::LoadTheme("crystalsvg"); // KDE3
|
| + }
|
| + default_themes[3] = IconTheme::LoadTheme("gnome");
|
| + } else { // Assume it's Gnome.
|
| + default_themes[1] = IconTheme::LoadTheme("gnome");
|
| + default_themes[2] = IconTheme::LoadTheme("crystalsvg"); // KDE3
|
| + default_themes[3] = IconTheme::LoadTheme("oxygen"); // KDE4
|
| + }
|
| + default_themes[4] = IconTheme::LoadTheme("hicolor");
|
| +}
|
| +
|
| +// Try to find an icon with the name |icon_name| that's |size| pixels.
|
| +FilePath LookupIconInDefaultTheme(const std::string& icon_name, int size) {
|
| + EnsureUpdated();
|
| + MimeUtilConstants* constants = Singleton<MimeUtilConstants>::get();
|
| + std::map<std::string, IconTheme*>* icon_themes = constants->icon_themes;
|
| + if (icon_themes->size() == 0) InitDefaultThemes();
|
| +
|
| + FilePath icon_path;
|
| + IconTheme** default_themes = constants->default_themes;
|
| + for (size_t i = 0; i < constants->kDefaultThemeNum; i++) {
|
| + if (default_themes[i]) {
|
| + icon_path = default_themes[i]->GetIconPath(icon_name, size, true);
|
| + if (!icon_path.empty())
|
| + return icon_path;
|
| + }
|
| + }
|
| + return LookupFallbackIcon(icon_name);
|
| +}
|
| +
|
| +} // namespace
|
| +
|
| +namespace mime_util {
|
| +
|
| +std::string GetFileMimeType(const std::string& file_path) {
|
| + return xdg_mime_get_mime_type_from_file_name(file_path.c_str());
|
| +}
|
| +
|
| +FilePath GetMimeIcon(const std::string& mime_type, size_t size) {
|
| + std::vector<std::string> icon_names;
|
| + std::string icon_name;
|
| + FilePath icon_file;
|
| +
|
| + const char* icon = xdg_mime_get_icon(mime_type.c_str());
|
| + icon_name = std::string(icon ? icon : "");
|
| + if (icon_name.length())
|
| + icon_names.push_back(icon_name);
|
| +
|
| + // For text/plain, try text-plain.
|
| + icon_name = mime_type;
|
| + for (size_t i = icon_name.find('/', 0); i != std::string::npos;
|
| + i = icon_name.find('/', i + 1)) {
|
| + icon_name[i] = '-';
|
| + }
|
| + icon_names.push_back(icon_name);
|
| + // Also try gnome-mime-text-plain.
|
| + icon_names.push_back("gnome-mime-" + icon_name);
|
| +
|
| + // Try generic name like text-x-generic.
|
| + icon_name = mime_type.substr(0, mime_type.find('/')) + "-x-generic";
|
| + icon_names.push_back(icon_name);
|
| +
|
| + // Last resort
|
| + icon_names.push_back("unknown");
|
| +
|
| + for (size_t i = 0; i < icon_names.size(); i++) {
|
| + if (icon_names[i][0] == '/') {
|
| + icon_file = FilePath(icon_names[i]);
|
| + if (file_util::PathExists(icon_file))
|
| + return icon_file;
|
| + } else {
|
| + icon_file = LookupIconInDefaultTheme(icon_names[i], size);
|
| + if (!icon_file.empty())
|
| + return icon_file;
|
| + }
|
| + }
|
| + return FilePath();
|
| +}
|
| +
|
| +} // namespace mime_util
|
|
|