Updating trunk VERSION from 2139.0 to 2140.0
[chromium-blink-merge.git] / chrome / renderer / web_apps.cc
blob2c9801ed0798836fd14fba78101677b249841393
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome/renderer/web_apps.h"
7 #include <string>
8 #include <vector>
10 #include "base/command_line.h"
11 #include "base/json/json_reader.h"
12 #include "base/strings/string16.h"
13 #include "base/strings/string_number_conversions.h"
14 #include "base/strings/string_split.h"
15 #include "base/strings/string_util.h"
16 #include "base/strings/utf_string_conversions.h"
17 #include "base/values.h"
18 #include "chrome/common/chrome_switches.h"
19 #include "chrome/common/web_application_info.h"
20 #include "third_party/WebKit/public/platform/WebString.h"
21 #include "third_party/WebKit/public/platform/WebURL.h"
22 #include "third_party/WebKit/public/web/WebDocument.h"
23 #include "third_party/WebKit/public/web/WebElement.h"
24 #include "third_party/WebKit/public/web/WebFrame.h"
25 #include "third_party/WebKit/public/web/WebNode.h"
26 #include "third_party/WebKit/public/web/WebNodeList.h"
27 #include "ui/gfx/size.h"
28 #include "url/gurl.h"
30 using blink::WebDocument;
31 using blink::WebElement;
32 using blink::WebFrame;
33 using blink::WebNode;
34 using blink::WebNodeList;
35 using blink::WebString;
37 namespace web_apps {
38 namespace {
40 // Sizes a single size (the width or height) from a 'sizes' attribute. A size
41 // matches must match the following regex: [1-9][0-9]*.
42 int ParseSingleIconSize(const base::string16& text) {
43 // Size must not start with 0, and be between 0 and 9.
44 if (text.empty() || !(text[0] >= L'1' && text[0] <= L'9'))
45 return 0;
47 // Make sure all chars are from 0-9.
48 for (size_t i = 1; i < text.length(); ++i) {
49 if (!(text[i] >= L'0' && text[i] <= L'9'))
50 return 0;
52 int output;
53 if (!base::StringToInt(text, &output))
54 return 0;
55 return output;
58 // Parses an icon size. An icon size must match the following regex:
59 // [1-9][0-9]*x[1-9][0-9]*.
60 // If the input couldn't be parsed, a size with a width/height == 0 is returned.
61 gfx::Size ParseIconSize(const base::string16& text) {
62 std::vector<base::string16> sizes;
63 base::SplitStringDontTrim(text, L'x', &sizes);
64 if (sizes.size() != 2)
65 return gfx::Size();
67 return gfx::Size(ParseSingleIconSize(sizes[0]),
68 ParseSingleIconSize(sizes[1]));
71 void AddInstallIcon(const WebElement& link,
72 std::vector<WebApplicationInfo::IconInfo>* icons) {
73 WebString href = link.getAttribute("href");
74 if (href.isNull() || href.isEmpty())
75 return;
77 // Get complete url.
78 GURL url = link.document().completeURL(href);
79 if (!url.is_valid())
80 return;
82 WebApplicationInfo::IconInfo icon_info;
83 bool is_any = false;
84 std::vector<gfx::Size> icon_sizes;
85 if (link.hasAttribute("sizes") &&
86 ParseIconSizes(link.getAttribute("sizes"), &icon_sizes, &is_any) &&
87 !is_any &&
88 icon_sizes.size() == 1) {
89 icon_info.width = icon_sizes[0].width();
90 icon_info.height = icon_sizes[0].height();
92 icon_info.url = url;
93 icons->push_back(icon_info);
96 } // namespace
98 bool ParseIconSizes(const base::string16& text,
99 std::vector<gfx::Size>* sizes,
100 bool* is_any) {
101 *is_any = false;
102 std::vector<base::string16> size_strings;
103 base::SplitStringAlongWhitespace(text, &size_strings);
104 for (size_t i = 0; i < size_strings.size(); ++i) {
105 if (EqualsASCII(size_strings[i], "any")) {
106 *is_any = true;
107 } else {
108 gfx::Size size = ParseIconSize(size_strings[i]);
109 if (size.width() <= 0 || size.height() <= 0)
110 return false; // Bogus size.
111 sizes->push_back(size);
114 if (*is_any && !sizes->empty()) {
115 // If is_any is true, it must occur by itself.
116 return false;
118 return (*is_any || !sizes->empty());
121 bool ParseWebAppFromWebDocument(WebFrame* frame,
122 WebApplicationInfo* app_info,
123 base::string16* error) {
124 WebDocument document = frame->document();
125 if (document.isNull())
126 return true;
128 WebElement head = document.head();
129 if (head.isNull())
130 return true;
132 GURL document_url = document.url();
133 WebNodeList children = head.childNodes();
134 for (unsigned i = 0; i < children.length(); ++i) {
135 WebNode child = children.item(i);
136 if (!child.isElementNode())
137 continue;
138 WebElement elem = child.to<WebElement>();
140 if (elem.hasHTMLTagName("link")) {
141 std::string rel = elem.getAttribute("rel").utf8();
142 // "rel" attribute may use either "icon" or "shortcut icon".
143 // see also
144 // <http://en.wikipedia.org/wiki/Favicon>
145 // <http://dev.w3.org/html5/spec/Overview.html#rel-icon>
147 // Streamlined Hosted Apps also support "apple-touch-icon" and
148 // "apple-touch-icon-precomposed".
149 if (LowerCaseEqualsASCII(rel, "icon") ||
150 LowerCaseEqualsASCII(rel, "shortcut icon") ||
151 (CommandLine::ForCurrentProcess()->
152 HasSwitch(switches::kEnableStreamlinedHostedApps) &&
153 (LowerCaseEqualsASCII(rel, "apple-touch-icon") ||
154 LowerCaseEqualsASCII(rel, "apple-touch-icon-precomposed")))) {
155 AddInstallIcon(elem, &app_info->icons);
157 } else if (elem.hasHTMLTagName("meta") && elem.hasAttribute("name")) {
158 std::string name = elem.getAttribute("name").utf8();
159 WebString content = elem.getAttribute("content");
160 if (name == "application-name") {
161 app_info->title = content;
162 } else if (name == "description") {
163 app_info->description = content;
164 } else if (name == "application-url") {
165 std::string url = content.utf8();
166 app_info->app_url = document_url.is_valid() ?
167 document_url.Resolve(url) : GURL(url);
168 if (!app_info->app_url.is_valid())
169 app_info->app_url = GURL();
174 return true;
177 } // namespace web_apps