MetaData rejectされました CFBundleDisplayNameを変更しましたが、 CFBundleNameも変更する必要がありそうです。 これで再審査だしてみます。
2015年5月29日金曜日
[iOS][Apple Watch] 錬金術師のレシピのApple Watch開発から申請まで3
ラベル:
Apple Watch,
iOS
2015年5月15日金曜日
[iOS][Apple Watch] 錬金術師のレシピのApple Watch開発から申請まで2
MetaData rejectされました プロジェクト名とアプリ名が違わない場合あまり起きないとおもわれる。 Rejectメッセージが丁寧になって、どうすればいいかまで書いてあってすごく助かります。 3.4 - App names in iTunes Connect and as displayed on a device should be similar, so as not to cause confusion に引っかかったようです。 ちゃんと、リストまで表示してくれています。 - Name on the App Store: Alchemist's Recipe - Name displayed on iPhone: Alchemist's - Name displayed on Apple Watch companion app: omoikane - Name displayed on Apple Watch: Alchemist's はじめ、 Name displayed on iPhoneは、長いので短くした表示名まで、合わせなくてはいけないようになったかと 焦りましたが、 これは、リリース時から通っているのでsimilarであると信じ - Name displayed on Apple Watch companion app: omoikane だけ修正をいれて審査にだしました。 InfoPlist.stringを追加して、CFBundleDisplayNameを定義(En,ja) extensionのアプリ名までちゃんとしないといけないと思いませんでした。 これで再審査だしてみます。 *プロジェクト名:omoikane(オモイカネ)は、知恵の神様です。
ラベル:
Apple Watch,
iOS
2015年5月6日水曜日
[iOS][Apple Watch] 錬金術師のレシピのApple Watch開発から申請まで
「錬金術師のレシピ」にApple Watch対応の開発から申請までの困ったこと(ISSUE)。
[開発]
1、iOS6対応の古いソースに追加してよいのかわからなかった。
追加可能だった。
2、xcode6.3.1を使用すると、apple watchがbuildできない。
IPHONEOS_DEPLOYMENT_TARGET = 8.3;を4箇所「8.2」にかえてやる必要がある
watch OSが8.2でiOSのバージョンより下なのが問題なのだろう。
3、Objectiv-Cで作成したが、WEBで検索するとサンプルがswift.
4、アプリ名のローカライズ
InfoPlist.stringでAppleWatchもアプリ名が指定できる
5、アイコンのサイズがpt単位
基本2倍のpxで、@3指定のものはptを3倍にすればOK.
形式でアルファチャンネルをいれないようにする。
対応したつもりでアップロードしたら、メールでアルファチャンネルはいっているよ。って送られてきた。
[申請]
1、Watch Kit,ExtensionともにAppidを作成して、プロビジョニングを作成する必要がある
開発段階だと気にしなくてよく、アーカイブもできるけど、いざアップロードしようとすると失敗する
2、5.5,4.7インチのスクリーンショットが、画面サイズに対応すると必須になった
とりあえず審査待ち状態になった。
ラベル:
Apple Watch,
iOS
2015年5月1日金曜日
HagaoitaのAndroid側のメモ
Ajaxでアクセスしたものに対して途中で、自分の好きなものを返す。
キャッシュされたデータ、写真とか返すとかに使われる手法。
(Android開発者登録したから、リリースしたいな。)
キャッシュされたデータ、写真とか返すとかに使われる手法。
(Android開発者登録したから、リリースしたいな。)
public class MainActivity extends ActionBarActivity {
@SuppressLint("NewApi")
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
WebView webView = (WebView)findViewById(R.id.webView);
webView.setWebViewClient(new MyWebViewClient());
webView.getSettings().setJavaScriptEnabled(true);
//これがないとlocalファイルでajaxできない
webView.getSettings().setAllowUniversalAccessFromFileURLs(true);
webView.loadUrl("file:///android_asset/index.html");
}
}
public class MyWebViewClient extends WebViewClient{
@Override
public WebResourceResponse shouldInterceptRequest(WebView view, String url) {
Log.d("#WEB", "aa111:"+url);
if(url.indexOf("7777")>0){
Log.d("#WEB", "aa111:IN");
InputStream bais = new ByteArrayInputStream("{'Hello':'test'}".getBytes());
WebResourceResponse res = new WebResourceResponse("application/json","UTF-8",bais);
return res;
}
return super.shouldInterceptRequest(view, url);
}
@Override
//本当はこっちがつかわれるはずなんだけど、この先でまだ、上のほうがつかわれている
public WebResourceResponse shouldInterceptRequest(WebView view, WebResourceRequest request) {
return super.shouldInterceptRequest(view, request);
}
}
2015年4月22日水曜日
環境構築してみた
sass:ある程度覚えた
compass:イマのところ、つかいどころなし
coffeescript: 楽しい部分もあるけど、微妙
gulpでタスクをいれるとすごくいい
compass:イマのところ、つかいどころなし
coffeescript: 楽しい部分もあるけど、微妙
gulpでタスクをいれるとすごくいい
browser-syncは、 変更ファイルがあるとその部分を配信更新してくれるので、 iphoneやシュミレータで、つくるときに、webkitやwebViewで一旦は、browser-syncのサーバでリモートでアクセスしながら作成できるので、形にできそうな気がします。その変更ソースは、iphoneの配信ソースそのままで良いので、リビルドなしである程度いけるというのが素晴らしい。
{
"name": "javascript",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"devDependencies": {
"browser-sync": "^2.6.4",
"gulp": "^3.8.11",
"gulp-autoprefixer": "^2.1.0",
"gulp-plumber": "^1.0.0",
"gulp-sass": "^1.3.3",
"gulp-uglify": "^1.2.0"
}
}
2015年4月12日日曜日
[ratchet] 次のステップ
一通りjsを確認したので、次のステップとして、 1,CSSをみる 2,POPサンプルのすごく簡易版をつくる 3,リストでスワイプしたら削除ボタンがでる とかをやってみたいと思う。
ラベル:
ratchet
[ratchet] push.jsをみる
今まで以上にながいけど、ひととおり目を通す。
- sessionStrageにデータを保持
- histをしっかり書き換える
もっと簡易なものをサンプルとしてかいてみると、もっと理解できそう。
/* ========================================================================
* Ratchet: push.js v2.0.2
* http://goratchet.com/components#push
* ========================================================================
* inspired by @defunkt's jquery.pjax.js
* Copyright 2014 Connor Sears
* Licensed under MIT (https://github.com/twbs/ratchet/blob/master/LICENSE)
* ======================================================================== */
/* global _gaq: true */
!(function () {
'use strict';
// no oprationとかの略かな?
var noop = function () {};
// Pushstate caching
// ==================
var isScrolling;
var maxCacheLength = 20;
/*
Web Storage
sessionStorage(ウィンドウやタブを閉じるまで)
localStorage(永続的)
*/
var cacheMapping = sessionStorage;
var domCache = {};
var transitionMap = {
slideIn : 'slide-out',
slideOut : 'slide-in',
fade : 'fade'
};
var bars = {
bartab : '.bar-tab',
barnav : '.bar-nav',
barfooter : '.bar-footer',
barheadersecondary : '.bar-header-secondary'
};
var cacheReplace = function (data, updates) {
PUSH.id = data.id;
if (updates) {
data = getCached(data.id);
}
cacheMapping[data.id] = JSON.stringify(data);
/*
現在の履歴エントリの URL または state オブジェクトを更新する
*/
window.history.replaceState(data.id, data.title, data.url);
/*
子ノードも含めてクローンしたものを保持
*/
domCache[data.id] = document.body.cloneNode(true);
};
/*
キャッシュにpushしている(データを追加している)
*/
var cachePush = function () {
var id = PUSH.id;
var cacheForwardStack = JSON.parse(cacheMapping.cacheForwardStack || '[]');
var cacheBackStack = JSON.parse(cacheMapping.cacheBackStack || '[]');
cacheBackStack.push(id);
/*
全部削除
*/
while (cacheForwardStack.length) {
delete cacheMapping[cacheForwardStack.shift()];
}
/*
指定数を超えたら、古いものから削除
*/
while (cacheBackStack.length > maxCacheLength) {
delete cacheMapping[cacheBackStack.shift()];
}
/*
遷移なしで履歴を追加
*/
window.history.pushState(null, '', cacheMapping[PUSH.id].url);
/*
作り直したものを保持
*/
cacheMapping.cacheForwardStack = JSON.stringify(cacheForwardStack);
cacheMapping.cacheBackStack = JSON.stringify(cacheBackStack);
};
/*
キャッシュにpopしている(データを(取り出して)削除している)
*/
var cachePop = function (id, direction) {
var forward = direction === 'forward';
var cacheForwardStack = JSON.parse(cacheMapping.cacheForwardStack || '[]');
var cacheBackStack = JSON.parse(cacheMapping.cacheBackStack || '[]');
var pushStack = forward ? cacheBackStack : cacheForwardStack;
var popStack = forward ? cacheForwardStack : cacheBackStack;
/*
すこし気になる部分、使う箇所によって意味がすごくかわる
*/
if (PUSH.id) {
pushStack.push(PUSH.id);
}
popStack.pop();
cacheMapping.cacheForwardStack = JSON.stringify(cacheForwardStack);
cacheMapping.cacheBackStack = JSON.stringify(cacheBackStack);
};
var getCached = function (id) {
return JSON.parse(cacheMapping[id] || null) || {};
};
var getTarget = function (e) {
//今までみてきたものと同じ、イベント発生部分をAタグで検索
//Aタグに何かをかぶせたりして、先にイベントを取っちゃうのをかんがえているのかな。
var target = findTarget(e.target);
if (!target ||
e.which > 1 ||
e.metaKey ||
e.ctrlKey ||
isScrolling ||
location.protocol !== target.protocol ||
location.host !== target.host ||
!target.hash && /#/.test(target.href) ||
target.hash && target.href.replace(target.hash, '') === location.href.replace(location.hash, '') ||
target.getAttribute('data-ignore') === 'push') { return; }
return target;
};
// Main event handlers (touchend, popstate)
// ==========================================
var touchend = function (e) {
var target = getTarget(e);
if (!target) {
return;
}
e.preventDefault();
PUSH({
url : target.href,
hash : target.hash,
timeout : target.getAttribute('data-timeout'),
transition : target.getAttribute('data-transition')
});
};
var popstate = function (e) {
var key;
var barElement;
var activeObj;
var activeDom;
var direction;
var transition;
var transitionFrom;
var transitionFromObj;
var id = e.state;
if (!id || !cacheMapping[id]) {
return;
}
direction = PUSH.id < id ? 'forward' : 'back';
cachePop(id, direction);
activeObj = getCached(id);
activeDom = domCache[id];
if (activeObj.title) {
document.title = activeObj.title;
}
if (direction === 'back') {
transitionFrom = JSON.parse(direction === 'back' ? cacheMapping.cacheForwardStack : cacheMapping.cacheBackStack);
transitionFromObj = getCached(transitionFrom[transitionFrom.length - 1]);
} else {
transitionFromObj = activeObj;
}
if (direction === 'back' && !transitionFromObj.id) {
return (PUSH.id = id);
}
transition = direction === 'back' ? transitionMap[transitionFromObj.transition] : transitionFromObj.transition;
if (!activeDom) {
return PUSH({
id : activeObj.id,
url : activeObj.url,
title : activeObj.title,
timeout : activeObj.timeout,
transition : transition,
ignorePush : true
});
}
if (transitionFromObj.transition) {
activeObj = extendWithDom(activeObj, '.content', activeDom.cloneNode(true));
for (key in bars) {
if (bars.hasOwnProperty(key)) {
barElement = document.querySelector(bars[key]);
if (activeObj[key]) {
swapContent(activeObj[key], barElement);
} else if (barElement) {
barElement.parentNode.removeChild(barElement);
}
}
}
}
swapContent(
(activeObj.contents || activeDom).cloneNode(true),
document.querySelector('.content'),
transition
);
PUSH.id = id;
document.body.offsetHeight; // force reflow to prevent scroll
};
// Core PUSH functionality
// =======================
var PUSH = function (options) {
var key;
var xhr = PUSH.xhr;
options.container = options.container || options.transition ? document.querySelector('.content') : document.body;
for (key in bars) {
if (bars.hasOwnProperty(key)) {
options[key] = options[key] || document.querySelector(bars[key]);
}
}
if (xhr && xhr.readyState < 4) {
xhr.onreadystatechange = noop;
//処理を止める
xhr.abort();
}
xhr = new XMLHttpRequest();
xhr.open('GET', options.url, true);
xhr.setRequestHeader('X-PUSH', 'true');
xhr.onreadystatechange = function () {
if (options._timeout) {
clearTimeout(options._timeout);
}
if (xhr.readyState === 4) {
xhr.status === 200 ? success(xhr, options) : failure(options.url);
}
};
if (!PUSH.id) {
cacheReplace({
id : +new Date(),
url : window.location.href,
title : document.title,
timeout : options.timeout,
transition : null
});
}
if (options.timeout) {
options._timeout = setTimeout(function () { xhr.abort('timeout'); }, options.timeout);
}
xhr.send();
if (xhr.readyState && !options.ignorePush) {
cachePush();
}
};
// Main XHR handlers
// =================
var success = function (xhr, options) {
var key;
var barElement;
var data = parseXHR(xhr, options);
if (!data.contents) {
return locationReplace(options.url);
}
if (data.title) {
document.title = data.title;
}
if (options.transition) {
for (key in bars) {
if (bars.hasOwnProperty(key)) {
barElement = document.querySelector(bars[key]);
if (data[key]) {
swapContent(data[key], barElement);
} else if (barElement) {
barElement.parentNode.removeChild(barElement);
}
}
}
}
swapContent(data.contents, options.container, options.transition, function () {
cacheReplace({
id : options.id || +new Date(),
url : data.url,
title : data.title,
timeout : options.timeout,
transition : options.transition
}, options.id);
triggerStateChange();
});
if (!options.ignorePush && window._gaq) {
_gaq.push(['_trackPageview']); // google analytics
}
if (!options.hash) {
return;
}
};
var failure = function (url) {
throw new Error('Could not get: ' + url);
};
// PUSH helpers
// ============
var swapContent = function (swap, container, transition, complete) {
var enter;
var containerDirection;
var swapDirection;
if (!transition) {
if (container) {
container.innerHTML = swap.innerHTML;
} else if (swap.classList.contains('content')) {
document.body.appendChild(swap);
} else {
document.body.insertBefore(swap, document.querySelector('.content'));
}
} else {
enter = /in$/.test(transition);
if (transition === 'fade') {
container.classList.add('in');
container.classList.add('fade');
swap.classList.add('fade');
}
if (/slide/.test(transition)) {
swap.classList.add('sliding-in', enter ? 'right' : 'left');
swap.classList.add('sliding');
container.classList.add('sliding');
}
container.parentNode.insertBefore(swap, container);
}
/*
以下遷移に合わせのアニメーション動作
*/
if (!transition) {
complete && complete();
}
if (transition === 'fade') {
container.offsetWidth; // force reflow
container.classList.remove('in');
var fadeContainerEnd = function () {
container.removeEventListener('webkitTransitionEnd', fadeContainerEnd);
swap.classList.add('in');
swap.addEventListener('webkitTransitionEnd', fadeSwapEnd);
};
var fadeSwapEnd = function () {
swap.removeEventListener('webkitTransitionEnd', fadeSwapEnd);
container.parentNode.removeChild(container);
swap.classList.remove('fade');
swap.classList.remove('in');
complete && complete();
};
container.addEventListener('webkitTransitionEnd', fadeContainerEnd);
}
if (/slide/.test(transition)) {
var slideEnd = function () {
swap.removeEventListener('webkitTransitionEnd', slideEnd);
swap.classList.remove('sliding', 'sliding-in');
swap.classList.remove(swapDirection);
container.parentNode.removeChild(container);
complete && complete();
};
container.offsetWidth; // force reflow
swapDirection = enter ? 'right' : 'left';
containerDirection = enter ? 'left' : 'right';
container.classList.add(containerDirection);
swap.classList.remove(swapDirection);
swap.addEventListener('webkitTransitionEnd', slideEnd);
}
};
/*
push完了通知
*/
var triggerStateChange = function () {
var e = new CustomEvent('push', {
detail: { state: getCached(PUSH.id) },
bubbles: true,
cancelable: true
});
window.dispatchEvent(e);
};
var findTarget = function (target) {
var i;
var toggles = document.querySelectorAll('a');
for (; target && target !== document; target = target.parentNode) {
for (i = toggles.length; i--;) {
if (toggles[i] === target) {
return target;
}
}
}
};
var locationReplace = function (url) {
window.history.replaceState(null, '', '#');
window.location.replace(url);
};
var extendWithDom = function (obj, fragment, dom) {
var i;
var result = {};
for (i in obj) {
if (obj.hasOwnProperty(i)) {
result[i] = obj[i];
}
}
Object.keys(bars).forEach(function (key) {
var el = dom.querySelector(bars[key]);
if (el) {
el.parentNode.removeChild(el);
}
result[key] = el;
});
result.contents = dom.querySelector(fragment);
return result;
};
var parseXHR = function (xhr, options) {
var head;
var body;
var data = {};
var responseText = xhr.responseText;
data.url = options.url;
if (!responseText) {
return data;
}
/*
取ってきたデータを変換している
*/
if (/]*>([\s\S.]*)<\/head>/i)[0];
body.innerHTML = responseText.match(/]*>([\s\S.]*)<\/body>/i)[0];
} else {
head = body = document.createElement('div');
head.innerHTML = responseText;
}
data.title = head.querySelector('title');
var text = 'innerText' in data.title ? 'innerText' : 'textContent';
data.title = data.title && data.title[text].trim();
if (options.transition) {
data = extendWithDom(data, '.content', body);
} else {
data.contents = body;
}
return data;
};
// Attach PUSH event handlers
// ==========================
window.addEventListener('touchstart', function () { isScrolling = false; });
window.addEventListener('touchmove', function () { isScrolling = true; });
window.addEventListener('touchend', touchend);
window.addEventListener('click', function (e) { if (getTarget(e)) {e.preventDefault();} });
window.addEventListener('popstate', popstate);
window.PUSH = PUSH;
}());
ラベル:
ratchet
登録:
コメント (Atom)