JavaScriptを色々あれこれしようとするがひたすら失敗を繰り返している
| Revision | 30 (tree) |
|---|---|
| Time | 2016-11-26 00:19:48 |
| Author | |
保存ファイル読み込み処理
| @@ -51,6 +51,12 @@ | ||
| 51 | 51 | } |
| 52 | 52 | |
| 53 | 53 | |
| 54 | + .DispObj { | |
| 55 | + display: inline-block; | |
| 56 | + position:absolute; | |
| 57 | + marginj:0; padding:0; | |
| 58 | + } | |
| 59 | + | |
| 54 | 60 | /* パレットグループ要素の定義 */ |
| 55 | 61 | .palgroup { |
| 56 | 62 | display: inline-block; |
| @@ -14,12 +14,12 @@ | ||
| 14 | 14 | <div id="DispField" data-objid="DisplayField" data-focus="true"> |
| 15 | 15 | </div> |
| 16 | 16 | <!-- 固定ボタン (選択要素の削除, 保存, …) --> |
| 17 | - <button type=button class="pal" data-objid="pal" data-palid="0" data-value="0xff000001">削除</button> | |
| 18 | - <button type=button class="pal" data-objid="pal" data-palid="1" data-value="0x00ff0001">設定保存</button> | |
| 19 | - <button type=button class="pal" data-objid="pal" data-palid="2" data-value="0x0000ff01">設定Drop</button> | |
| 20 | - <button type=button class="pal" data-objid="pal" data-palid="3" data-value="0xffffff01">アニメ停止</button> | |
| 21 | - <button type=button class="pal" data-objid="pal" data-palid="4" data-value="0x00000000">button5</button> | |
| 22 | - <a href="#" id="save" download="savefilename.txt">画像保存</a><br /> | |
| 17 | + <button type=button class="pal" data-objid="pal" data-palid="0" title="フォーカス中の要素を削除します">削除</button> | |
| 18 | + <button type=button class="pal" data-objid="pal" data-palid="1" title="現在のパレット設定を保存します">設定保存</button> | |
| 19 | + <button type=button class="pal" data-objid="pal" data-palid="2" title="設定画面を表示します">設定Drop</button> | |
| 20 | + <button type=button class="pal" data-objid="pal" data-palid="3" title="アニメーションを停止します">アニメ停止</button> | |
| 21 | + <button type=button class="pal" data-objid="pal" data-palid="4" >button5</button> | |
| 22 | + <a href="#" id="save" download="DragDraw.dd" title="DragDraw.ddという名前でダウンロードフォルダに保存します">画像保存</a><br /> | |
| 23 | 23 | |
| 24 | 24 | <!-- Palette --> |
| 25 | 25 | <style id="stylepalbg"> |
| @@ -158,7 +158,7 @@ | ||
| 158 | 158 | } |
| 159 | 159 | </style> |
| 160 | 160 | |
| 161 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup0"> | |
| 161 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup0" title="背景の色や画像を設定するパレットです"> | |
| 162 | 162 | <legend class="pallegend"> Background Palette </legend> |
| 163 | 163 | <div class="palbg palbg0" data-objid="palbg0" data-palid="0"></div> |
| 164 | 164 | <div class="palbg palbg1" data-objid="palbg1" data-palid="1"></div> |
| @@ -168,7 +168,7 @@ | ||
| 168 | 168 | </fieldset> |
| 169 | 169 | <br /> |
| 170 | 170 | |
| 171 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup1"> | |
| 171 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup1" title="テキストの色を設定するパレットです"> | |
| 172 | 172 | <legend class="pallegend"> Text Color Palette </legend> |
| 173 | 173 | <div class="paltxtcol paltxtcol0" data-objid="paltxtcol0" data-palid="0">あ</div> |
| 174 | 174 | <div class="paltxtcol paltxtcol1" data-objid="paltxtcol1" data-palid="1">あ</div> |
| @@ -179,7 +179,7 @@ | ||
| 179 | 179 | |
| 180 | 180 | <br /> |
| 181 | 181 | |
| 182 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup2"> | |
| 182 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup2" title="動きを設定するパレットです"> | |
| 183 | 183 | <legend class="pallegend"> Action Palette </legend> |
| 184 | 184 | <div class="palact palact0" data-objid="palact0" data-palid="0"></div> |
| 185 | 185 | <div class="palact palact1" data-objid="palact1" data-palid="1"></div> |
| @@ -10,9 +10,12 @@ | ||
| 10 | 10 | function logQuiet(str) { |
| 11 | 11 | // 処理なし |
| 12 | 12 | }; |
| 13 | +function isStoreApp() { | |
| 14 | + // ★識別方法を知りたい…… | |
| 15 | + return true; | |
| 16 | +}; | |
| 13 | 17 | |
| 14 | 18 | |
| 15 | - | |
| 16 | 19 | function rectData( left, top, right, bottom ) { |
| 17 | 20 | this.setData( left, top, right, bottom ); |
| 18 | 21 | }; |
| @@ -42,6 +45,7 @@ | ||
| 42 | 45 | |
| 43 | 46 | // Image fileを読み込み、その内容を対象CSSに設定 |
| 44 | 47 | // CSSには "'cssstr1' + fileBlob + 'cssstr2'" が設定される |
| 48 | +// image/text共に読み込んだ内容をそのままCSS定義にする | |
| 45 | 49 | function setImageFileToCSS( |
| 46 | 50 | ele, // CSS書き換え対象Element |
| 47 | 51 | num, // 書き換える CSS要素番号 (ele.sheet.cssRules[n] ← このnの値) |
| @@ -61,7 +65,7 @@ | ||
| 61 | 65 | num, // 書き換える CSS要素番号 (ele.sheet.cssRules[n] ← このnの値) |
| 62 | 66 | cssstr1, // CSS書き換え前半文字 |
| 63 | 67 | cssstr2, // CSS書き換え後半文字 |
| 64 | - file // Textファイルを指している file object | |
| 68 | + file // CSS定義が1つだけ入った Textファイルを指している file object | |
| 65 | 69 | ) { |
| 66 | 70 | // drop fileの読み込み |
| 67 | 71 | let reader = new FileReader(); |
| @@ -77,8 +81,122 @@ | ||
| 77 | 81 | ele.sheet.insertRule(csstext, num); |
| 78 | 82 | }; |
| 79 | 83 | |
| 84 | +// CSSに定義をセット | |
| 85 | +function setStrToCSS( | |
| 86 | + ele, // CSS設定先 style object | |
| 87 | + num, // 設定対象添え字 (ele.sheet.cssRule[n] の nの値) | |
| 88 | + str, // 設定する文字列 | |
| 89 | + reg // [option] 抜き出し用 正規表現, または null | |
| 90 | +) { | |
| 91 | + if (null == ele | |
| 92 | + || null == str | |
| 93 | + || ele.sheet.cssRules.length <= num | |
| 94 | + || num < 0) { | |
| 95 | + console.error('CSS定義追加に失敗 setStrToCSS() パラメータエラー'); | |
| 96 | + console.error('setStrToCSS params ele=' + ele + ', num=' + num + ', ele.cssRules num=' + ((null != ele) ? ele.sheet.cssRules.length : 'null') + ', str=' + str); | |
| 97 | + return false; | |
| 98 | + } | |
| 80 | 99 | |
| 100 | + let addText = str; | |
| 101 | + if (null != reg) { | |
| 102 | + let matchstr = str.match(reg); | |
| 103 | + if (null == matchstr) { | |
| 104 | + console.error('CSS定義追加に失敗 setStrToCSS() 追加文字列エラー'); | |
| 105 | + console.error('setStrToCSS match result=' + matchstr + ', reg=' + reg + ', str=' + str); | |
| 106 | + return false ; | |
| 107 | + } | |
| 81 | 108 | |
| 109 | + addText = matchstr[0]; | |
| 110 | + } | |
| 111 | + ele.sheet.deleteRule(num); | |
| 112 | + ele.sheet.insertRule(addText, num); | |
| 113 | + | |
| 114 | + return true; | |
| 115 | +} | |
| 116 | + | |
| 117 | + | |
| 118 | +// 引数elementに設定されたstylesheetから 全 | |
| 119 | +// "animation定義を含むcss", "対応したkeyframe" を取得する | |
| 120 | +// (@keyframe定義がcssTextから取得できないために作成。 | |
| 121 | +// keyframeはcssのtextではないらしい。css仕様を作っている奴らは馬鹿か!) | |
| 122 | +function getAnimationCssFromElement(ele) { | |
| 123 | + let animationrule = null; | |
| 124 | + let keyframename = null; | |
| 125 | + let keyframerule = null; | |
| 126 | + let outstr = ""; | |
| 127 | + | |
| 128 | + debuglog('アクションパレット情報(CSS) 取得開始 getAnimationCssFromElement() ***************'); | |
| 129 | + for (let i = 0; i < ele.sheet.cssRules.length; i++) { | |
| 130 | + animationrule = ele.sheet.cssRules[i]; | |
| 131 | + if (window.CSSRule.STYLE_RULE == animationrule.type) { | |
| 132 | + // animation定義文字列から keyframe nameを探して取得 | |
| 133 | + keyframename = getNameOfKeyframe(animationrule.style.cssText); | |
| 134 | + if (null != keyframename) { | |
| 135 | + // document内の全style sheetから引数のkeyframe定義を取得 | |
| 136 | + keyframerule = getKeyframeRule(keyframename); | |
| 137 | + } | |
| 138 | + // 指定keyframe定義がない or そもそもanimation定義でない or animation定義syntaxerr の場合は | |
| 139 | + // keyframe領域確保のためダミー定義を返す | |
| 140 | + if (null == keyframerule) { | |
| 141 | + console.error('animation ' + animationrule.style.cssText + 'に対応するkeyframe ' + keyframename + ' が見つからないため、ダミーを設定'); | |
| 142 | + keyframerule = '@keyframes dummy { to { color: black;} }'; | |
| 143 | + } | |
| 144 | + | |
| 145 | + // animation定義を復元 | |
| 146 | + let animationcss = animationrule.selectorText + ' {\r\n' + animationrule.style.cssText + '\r\n}'; | |
| 147 | + debuglog('パレット情報1件追加 ele.sheet.cssRules[' + i + ']\n' + animationcss + '\n' + keyframerule.cssText); | |
| 148 | + outstr += (animationcss + '\r\n' + keyframerule.cssText + '\r\n'); | |
| 149 | + } | |
| 150 | + } | |
| 151 | + debuglog('****************************************************'); | |
| 152 | + | |
| 153 | + return outstr; | |
| 154 | +}; | |
| 155 | +// 引数cssstrから animation-name(@keyframeで付けられる) を取得 | |
| 156 | +function getNameOfKeyframe( | |
| 157 | + cssstr // animation または animation-nameが定義された css文字列 | |
| 158 | +) { | |
| 159 | + // animation-nameでも shorthandでも keyframe nameはparamの先頭固定 | |
| 160 | + let name = null; | |
| 161 | + name = cssstr.match(/animation-name *: *(.+?) /); | |
| 162 | + if (null == name) { | |
| 163 | + name = cssstr.match(/animation *: *(.+?) /); | |
| 164 | + if (null == name) { | |
| 165 | + console.error('文字列から @keyframe名の検索に失敗 getNameOfKeyframe() str=' + cssstr); | |
| 166 | + return null; | |
| 167 | + } | |
| 168 | + } | |
| 169 | + return name[1]; | |
| 170 | +}; | |
| 171 | + | |
| 172 | +// 現在のdocumentに設定されている 全Style Sheetから | |
| 173 | +// ruleNameの名前がついた @keyframe検索 | |
| 174 | +// (Stack Overflowからのもらいもの) | |
| 175 | +function getKeyframeRule(ruleName) { | |
| 176 | + var ss = document.styleSheets, | |
| 177 | + rule, i, x; | |
| 178 | + | |
| 179 | + for (i = 0; i < ss.length; ++i) { | |
| 180 | + if (ss[i].cssRules) { | |
| 181 | + // loop through all the rules | |
| 182 | + | |
| 183 | + for (x = ss[i].cssRules.length - 1; x >= 0; x--) { | |
| 184 | + rule = ss[i].cssRules[x]; | |
| 185 | + if ((rule.type === window.CSSRule.KEYFRAMES_RULE || rule.type === window.CSSRule.WEBKIT_KEYFRAMES_RULE) && rule.name === ruleName) { | |
| 186 | + return rule; | |
| 187 | + } | |
| 188 | + } | |
| 189 | + } | |
| 190 | + } | |
| 191 | + return null; | |
| 192 | +} | |
| 193 | + | |
| 194 | + | |
| 195 | + | |
| 196 | + | |
| 197 | + | |
| 198 | + | |
| 199 | + | |
| 82 | 200 | // Drop fileから file objectを取得 (同期処理) |
| 83 | 201 | // 複数file 非対応。text or image がヒットした時点でcallbackを呼び出す |
| 84 | 202 | // callback書式(text,image): function(blob) |
| @@ -80,7 +80,6 @@ | ||
| 80 | 80 | // requestAnimationFrameのcallbackから呼び出される |
| 81 | 81 | render() { |
| 82 | 82 | while (0 < this.renderFunc.size()) { |
| 83 | - this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 84 | 83 | let renderFunction = this.renderFunc.dequeue(); |
| 85 | 84 | renderFunction(); |
| 86 | 85 | } |
| @@ -116,6 +115,7 @@ | ||
| 116 | 115 | // 画面に影響しない範囲で作成できるところまで Dom objectを作成する |
| 117 | 116 | let elType = (null != type.match(/imagebox/)) ? 'img' : 'div'; |
| 118 | 117 | this.DOMobject = document.createElement(elType); |
| 118 | + this.DOMobject.classList.add('DispObj'); | |
| 119 | 119 | this.DOMobject.dataset.objid = this.ObjID; |
| 120 | 120 | this.DOMobject.dataset.orgwidth = this.originalWidth; // transform: scaleの基準サイズ |
| 121 | 121 | this.DOMobject.dataset.orgheight = this.originalHeight; |
| @@ -185,13 +185,12 @@ | ||
| 185 | 185 | return new rectData(0, 0, destW, destH); |
| 186 | 186 | } |
| 187 | 187 | renderCreate() { |
| 188 | - this.DOMobject.style.position = "absolute"; // 決め打ち | |
| 188 | +// this.DOMobject.style.position = "absolute"; // DispObj CSS classで指定 | |
| 189 | 189 | this.DOMobject.style.left = this.x.toString() + "px"; |
| 190 | 190 | this.DOMobject.style.top = this.y.toString() + "px"; |
| 191 | 191 | this.DOMobject.style.width = this.width.toString() + "px"; |
| 192 | 192 | this.DOMobject.style.height = this.height.toString() + "px"; |
| 193 | 193 | this.DOMobject.style.border = "1px solid black"; // 指定するI/Fが必要 |
| 194 | -// this.DOMobject.style.backgroundColor = "#cdcdcd"; | |
| 195 | 194 | if (null != this.text) this.DOMobject.innerHTML = this.text; |
| 196 | 195 | |
| 197 | 196 | document.getElementById('DispField').appendChild(this.DOMobject); |
| @@ -203,6 +202,28 @@ | ||
| 203 | 202 | + ", border=" + this.DOMobject.style.border); |
| 204 | 203 | } |
| 205 | 204 | |
| 205 | + createFromElement(ele) { | |
| 206 | + this.debuglog('createFromElement() ele=' + ele); | |
| 207 | + | |
| 208 | + this.DOMobject = ele; | |
| 209 | + this.type = (null != ele.nodeName.match(/IMG/)) ? 'imagebox' : 'textbox'; | |
| 210 | + this.x = parseInt(ele.style.left); | |
| 211 | + this.y = parseInt(ele.style.top); | |
| 212 | + this.width = parseInt(ele.style.width); | |
| 213 | + this.height = parseInt(ele.style.height); | |
| 214 | + this.originalWidth = ele.dataset.orgwidth; | |
| 215 | + this.originalHeight = ele.dataset.orgheight; | |
| 216 | + this.ObjID = ele.dataset.objid; | |
| 217 | + this.degree = ele.dataset.degree; | |
| 218 | + this.DOMobject.addEventListener('mousedown', this.onMouseDown.bind(this), false); | |
| 219 | + | |
| 220 | + // 画像・テキストは既にelementに読み込み済み | |
| 221 | + // treeにも入っており、更新する必要もないのでこれで完了 | |
| 222 | + } | |
| 223 | + | |
| 224 | + | |
| 225 | + | |
| 226 | + | |
| 206 | 227 | // DIV box 移動 (移動先確定後の移動) |
| 207 | 228 | // ※ Drag中の表示は別で行っている |
| 208 | 229 | movebox(x, y) { |
| @@ -268,6 +289,8 @@ | ||
| 268 | 289 | } |
| 269 | 290 | // Mouse eventをappAreaからScalerに渡してもらうように設定 |
| 270 | 291 | window.appArea.setMouseEventObj(this.mouseMove.bind(this), this.mouseUp.bind(this)); |
| 292 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 293 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 271 | 294 | |
| 272 | 295 | |
| 273 | 296 | // Focus設定 |
| @@ -281,6 +304,9 @@ | ||
| 281 | 304 | |
| 282 | 305 | } |
| 283 | 306 | mouseMove(evt) { |
| 307 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 308 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 309 | + | |
| 284 | 310 | // 画像の仮移動 |
| 285 | 311 | this.endDragX = evt.pageX; |
| 286 | 312 | this.endDragY = evt.pageY; |
| @@ -302,6 +328,8 @@ | ||
| 302 | 328 | this.debuglog("mouseUp"); |
| 303 | 329 | // Mouse event処理対象をクリア |
| 304 | 330 | window.appArea.setMouseEventObj(null, null); |
| 331 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 332 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 305 | 333 | |
| 306 | 334 | |
| 307 | 335 | // 移動元として座標を保持 |
| @@ -39,7 +39,8 @@ | ||
| 39 | 39 | let styleaction = document.getElementById('stylepalact'); |
| 40 | 40 | let cssbg = stylebg.sheet.cssText; |
| 41 | 41 | let csstextcol = styletextcol.sheet.cssText; |
| 42 | - let cssaction = styleaction.sheet.cssText; | |
| 42 | + let cssaction = getAnimationCssFromElement(styleaction); | |
| 43 | + | |
| 43 | 44 | // 最終出力 |
| 44 | 45 | let savestr = '<style id="stylepalbg">' + cssbg + '</style><style id="stylepaltxtcol">' + csstextcol + '</style><style id="stylepalact">' + cssaction + '</style>' + displayInfo; |
| 45 | 46 |
| @@ -49,10 +50,11 @@ | ||
| 49 | 50 | this.DOMobject.innerHTML = text; |
| 50 | 51 | } |
| 51 | 52 | |
| 53 | + | |
| 54 | + | |
| 52 | 55 | // ブラウザ上への描画 ------------------------- |
| 53 | 56 | render() { |
| 54 | 57 | while (0 < this.renderFunc.size()) { |
| 55 | - // this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 56 | 58 | let renderFunction = this.renderFunc.dequeue(); |
| 57 | 59 | renderFunction(); |
| 58 | 60 | } |
| @@ -82,6 +82,29 @@ | ||
| 82 | 82 | } |
| 83 | 83 | |
| 84 | 84 | |
| 85 | + // 復元されたDOMに合わせて DispObjを作成する | |
| 86 | + // File Load 処理用 | |
| 87 | + adjustEnvToDOM() { | |
| 88 | + this.clear(); | |
| 89 | + | |
| 90 | + // 復元されたDispObj elementを取得 | |
| 91 | + let listEle = document.getElementsByClassName('DispObj'); | |
| 92 | + | |
| 93 | + this.debuglog('読み込み処理 DOM element設定から内部instanceを作成\nElement=' + listEle.length); | |
| 94 | + | |
| 95 | + let ele = null; | |
| 96 | + for (let i = 0; i < listEle.length; i++) { | |
| 97 | + ele = listEle[i]; | |
| 98 | + this.debuglog('読み込み処理 Element=' + (i+1) + '/' + listEle.length + '\ntype=' + ele.nodeName + ', objid=' + ele.dataset.objid); | |
| 99 | + let dispObj = new CDispObj(); | |
| 100 | + dispObj.createFromElement(ele); | |
| 101 | + | |
| 102 | + this.ObjIDLen = parseInt(ele.dataset.objid) + 1; | |
| 103 | + this.ObjIDarray[this.ObjIDLen - 1] = dispObj; | |
| 104 | + } | |
| 105 | + } | |
| 106 | + | |
| 107 | + | |
| 85 | 108 | // Command処理 ---------------------- |
| 86 | 109 | createDispObj(type, rect, opt) { |
| 87 | 110 | this.debuglog("createDispObj"); |
| @@ -33,7 +33,6 @@ | ||
| 33 | 33 | // requestAnimationFrameからの呼び出しを想定 |
| 34 | 34 | render() { |
| 35 | 35 | while (0 < this.renderFunc.size()) { |
| 36 | - // this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 37 | 36 | let renderFunction = this.renderFunc.dequeue(); |
| 38 | 37 | renderFunction(); |
| 39 | 38 | } |
| @@ -28,7 +28,7 @@ | ||
| 28 | 28 | |
| 29 | 29 | // Mouse eventを処理するObj |
| 30 | 30 | setMouseEventObj(mouseMove, mouseUp) { |
| 31 | - this.debuglog('setMouseEventObj'); | |
| 31 | + this.debuglog('setMouseEventObj callbackMove=' + ((null != mouseMove)? mouseMove.name: 'null') + ', callbackUp=' + ((null != mouseUp)? mouseUp.name: 'null')); | |
| 32 | 32 | |
| 33 | 33 | this.mouseMoveFunc = mouseMove; |
| 34 | 34 | this.mouseUpFunc = mouseUp; |
| @@ -65,7 +65,7 @@ | ||
| 65 | 65 | file // 画像を指しているfile object |
| 66 | 66 | ) { |
| 67 | 67 | let cssstr1 = '.palbg' + id + ' { background-image: url('; |
| 68 | - let cssstr2 = '); }'; | |
| 68 | + let cssstr2 = '); background-size: cover; }'; | |
| 69 | 69 | setImageFileToCSS(this.stylebg, id, cssstr1, cssstr2, file); |
| 70 | 70 | } |
| 71 | 71 |
| @@ -177,8 +177,11 @@ | ||
| 177 | 177 | // Mouse event |
| 178 | 178 | // 自elementをDragし、Drop時にTarget elementに自分のCSS classを適用する |
| 179 | 179 | onMouseDown(evt) { |
| 180 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 181 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 182 | + | |
| 180 | 183 | // Mouse eventをappAreaからCSSPalに渡してもらうように設定 |
| 181 | - window.appArea.setMouseEventObj(this.mouseMove.bind(this), null); | |
| 184 | + window.appArea.setMouseEventObj(this.mouseMove.bind(this), this.mouseUp.bind(this)); | |
| 182 | 185 | |
| 183 | 186 | // Drag対象を保持 |
| 184 | 187 | this.draggingDOM = evt.target; |
| @@ -191,6 +194,8 @@ | ||
| 191 | 194 | } |
| 192 | 195 | mouseMove(evt) { |
| 193 | 196 | if (null == this.draggingDOM) return; |
| 197 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 198 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 194 | 199 | |
| 195 | 200 | // 画像の仮移動 |
| 196 | 201 | this.endDragX = evt.pageX; |
| @@ -206,6 +211,8 @@ | ||
| 206 | 211 | //this.renderMoveDragging(moveX, moveY); |
| 207 | 212 | } |
| 208 | 213 | renderMoveDragging(x, y) { |
| 214 | + if (null == this.draggingDOM) return; | |
| 215 | + | |
| 209 | 216 | this.draggingDOM.style.left = x + "px"; |
| 210 | 217 | this.draggingDOM.style.top = y + "px"; |
| 211 | 218 | if( 1.0 == this.draggingDOM.style.opacity ) |
| @@ -212,6 +219,10 @@ | ||
| 212 | 219 | this.draggingDOM.style.opacity = 0.4; |
| 213 | 220 | } |
| 214 | 221 | mouseUp(evt) { |
| 222 | + if (null == this.draggingDOM) return; | |
| 223 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 224 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 225 | + | |
| 215 | 226 | let palobjid = this.draggingDOM.dataset.objid; |
| 216 | 227 | let palid = this.draggingDOM.dataset.palid; |
| 217 | 228 |
| @@ -56,6 +56,8 @@ | ||
| 56 | 56 | onMouseDown(evt) { |
| 57 | 57 | // Mouse eventをappAreaからCSSPalに渡してもらうように設定 |
| 58 | 58 | window.appArea.setMouseEventObj(this.mouseMove.bind(this), null); |
| 59 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 60 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 59 | 61 | |
| 60 | 62 | // Drag対象を保持 |
| 61 | 63 | this.draggingDOM = evt.target; |
| @@ -68,6 +70,8 @@ | ||
| 68 | 70 | } |
| 69 | 71 | mouseMove(evt) { |
| 70 | 72 | if (null == this.draggingDOM) return; |
| 73 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 74 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 71 | 75 | |
| 72 | 76 | // 画像の仮移動 |
| 73 | 77 | this.endDragX = evt.pageX; |
| @@ -90,6 +94,10 @@ | ||
| 90 | 94 | this.draggingDOM.style.top = y + "px"; |
| 91 | 95 | } |
| 92 | 96 | mouseUp(evt) { |
| 97 | + if (null == this.draggingDOM) return; | |
| 98 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 99 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 100 | + | |
| 93 | 101 | // Drag中 element情報をクリア |
| 94 | 102 | this.draggingDOM = null; |
| 95 | 103 | // Mouse event callback設定をクリア |
| @@ -51,6 +51,12 @@ | ||
| 51 | 51 | } |
| 52 | 52 | |
| 53 | 53 | |
| 54 | + .DispObj { | |
| 55 | + display: inline-block; | |
| 56 | + position:absolute; | |
| 57 | + marginj:0; padding:0; | |
| 58 | + } | |
| 59 | + | |
| 54 | 60 | /* パレットグループ要素の定義 */ |
| 55 | 61 | .palgroup { |
| 56 | 62 | display: inline-block; |
| @@ -14,12 +14,12 @@ | ||
| 14 | 14 | <div id="DispField" data-objid="DisplayField" data-focus="true"> |
| 15 | 15 | </div> |
| 16 | 16 | <!-- 固定ボタン (選択要素の削除, 保存, …) --> |
| 17 | - <button type=button class="pal" data-objid="pal" data-palid="0" data-value="0xff000001">削除</button> | |
| 18 | - <button type=button class="pal" data-objid="pal" data-palid="1" data-value="0x00ff0001">設定保存</button> | |
| 19 | - <button type=button class="pal" data-objid="pal" data-palid="2" data-value="0x0000ff01">設定Drop</button> | |
| 20 | - <button type=button class="pal" data-objid="pal" data-palid="3" data-value="0xffffff01">アニメ停止</button> | |
| 21 | - <button type=button class="pal" data-objid="pal" data-palid="4" data-value="0x00000000">button5</button> | |
| 22 | - <a href="#" id="save" download="savefilename.txt">画像保存</a><br /> | |
| 17 | + <button type=button class="pal" data-objid="pal" data-palid="0" title="フォーカス中の要素を削除します">削除</button> | |
| 18 | + <button type=button class="pal" data-objid="pal" data-palid="1" title="現在のパレット設定を保存します">設定保存</button> | |
| 19 | + <button type=button class="pal" data-objid="pal" data-palid="2" title="設定画面を表示します">設定Drop</button> | |
| 20 | + <button type=button class="pal" data-objid="pal" data-palid="3" title="アニメーションを停止します">アニメ停止</button> | |
| 21 | + <button type=button class="pal" data-objid="pal" data-palid="4" >button5</button> | |
| 22 | + <a href="#" id="save" download="DragDraw.dd" title="DragDraw.ddという名前でダウンロードフォルダに保存します">画像保存</a><br /> | |
| 23 | 23 | |
| 24 | 24 | <!-- Palette --> |
| 25 | 25 | <style id="stylepalbg"> |
| @@ -158,7 +158,7 @@ | ||
| 158 | 158 | } |
| 159 | 159 | </style> |
| 160 | 160 | |
| 161 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup0"> | |
| 161 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup0" title="背景の色や画像を設定するパレットです"> | |
| 162 | 162 | <legend class="pallegend"> Background Palette </legend> |
| 163 | 163 | <div class="palbg palbg0" data-objid="palbg0" data-palid="0"></div> |
| 164 | 164 | <div class="palbg palbg1" data-objid="palbg1" data-palid="1"></div> |
| @@ -168,7 +168,7 @@ | ||
| 168 | 168 | </fieldset> |
| 169 | 169 | <br /> |
| 170 | 170 | |
| 171 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup1"> | |
| 171 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup1" title="テキストの色を設定するパレットです"> | |
| 172 | 172 | <legend class="pallegend"> Text Color Palette </legend> |
| 173 | 173 | <div class="paltxtcol paltxtcol0" data-objid="paltxtcol0" data-palid="0">あ</div> |
| 174 | 174 | <div class="paltxtcol paltxtcol1" data-objid="paltxtcol1" data-palid="1">あ</div> |
| @@ -179,7 +179,7 @@ | ||
| 179 | 179 | |
| 180 | 180 | <br /> |
| 181 | 181 | |
| 182 | - <fieldset class="palgroup palgrouphorizon" data-objid="palgroup2"> | |
| 182 | + <fieldset class="palgroup palgrouphorizon" data-objid="palgroup2" title="動きを設定するパレットです"> | |
| 183 | 183 | <legend class="pallegend"> Action Palette </legend> |
| 184 | 184 | <div class="palact palact0" data-objid="palact0" data-palid="0"></div> |
| 185 | 185 | <div class="palact palact1" data-objid="palact1" data-palid="1"></div> |
| @@ -10,9 +10,12 @@ | ||
| 10 | 10 | function logQuiet(str) { |
| 11 | 11 | // 処理なし |
| 12 | 12 | }; |
| 13 | +function isStoreApp() { | |
| 14 | + // ★識別方法を知りたい…… | |
| 15 | + return true; | |
| 16 | +}; | |
| 13 | 17 | |
| 14 | 18 | |
| 15 | - | |
| 16 | 19 | function rectData( left, top, right, bottom ) { |
| 17 | 20 | this.setData( left, top, right, bottom ); |
| 18 | 21 | }; |
| @@ -42,6 +45,7 @@ | ||
| 42 | 45 | |
| 43 | 46 | // Image fileを読み込み、その内容を対象CSSに設定 |
| 44 | 47 | // CSSには "'cssstr1' + fileBlob + 'cssstr2'" が設定される |
| 48 | +// image/text共に読み込んだ内容をそのままCSS定義にする | |
| 45 | 49 | function setImageFileToCSS( |
| 46 | 50 | ele, // CSS書き換え対象Element |
| 47 | 51 | num, // 書き換える CSS要素番号 (ele.sheet.cssRules[n] ← このnの値) |
| @@ -61,7 +65,7 @@ | ||
| 61 | 65 | num, // 書き換える CSS要素番号 (ele.sheet.cssRules[n] ← このnの値) |
| 62 | 66 | cssstr1, // CSS書き換え前半文字 |
| 63 | 67 | cssstr2, // CSS書き換え後半文字 |
| 64 | - file // Textファイルを指している file object | |
| 68 | + file // CSS定義が1つだけ入った Textファイルを指している file object | |
| 65 | 69 | ) { |
| 66 | 70 | // drop fileの読み込み |
| 67 | 71 | let reader = new FileReader(); |
| @@ -77,8 +81,122 @@ | ||
| 77 | 81 | ele.sheet.insertRule(csstext, num); |
| 78 | 82 | }; |
| 79 | 83 | |
| 84 | +// CSSに定義をセット | |
| 85 | +function setStrToCSS( | |
| 86 | + ele, // CSS設定先 style object | |
| 87 | + num, // 設定対象添え字 (ele.sheet.cssRule[n] の nの値) | |
| 88 | + str, // 設定する文字列 | |
| 89 | + reg // [option] 抜き出し用 正規表現, または null | |
| 90 | +) { | |
| 91 | + if (null == ele | |
| 92 | + || null == str | |
| 93 | + || ele.sheet.cssRules.length <= num | |
| 94 | + || num < 0) { | |
| 95 | + console.error('CSS定義追加に失敗 setStrToCSS() パラメータエラー'); | |
| 96 | + console.error('setStrToCSS params ele=' + ele + ', num=' + num + ', ele.cssRules num=' + ((null != ele) ? ele.sheet.cssRules.length : 'null') + ', str=' + str); | |
| 97 | + return false; | |
| 98 | + } | |
| 80 | 99 | |
| 100 | + let addText = str; | |
| 101 | + if (null != reg) { | |
| 102 | + let matchstr = str.match(reg); | |
| 103 | + if (null == matchstr) { | |
| 104 | + console.error('CSS定義追加に失敗 setStrToCSS() 追加文字列エラー'); | |
| 105 | + console.error('setStrToCSS match result=' + matchstr + ', reg=' + reg + ', str=' + str); | |
| 106 | + return false ; | |
| 107 | + } | |
| 81 | 108 | |
| 109 | + addText = matchstr[0]; | |
| 110 | + } | |
| 111 | + ele.sheet.deleteRule(num); | |
| 112 | + ele.sheet.insertRule(addText, num); | |
| 113 | + | |
| 114 | + return true; | |
| 115 | +} | |
| 116 | + | |
| 117 | + | |
| 118 | +// 引数elementに設定されたstylesheetから 全 | |
| 119 | +// "animation定義を含むcss", "対応したkeyframe" を取得する | |
| 120 | +// (@keyframe定義がcssTextから取得できないために作成。 | |
| 121 | +// keyframeはcssのtextではないらしい。css仕様を作っている奴らは馬鹿か!) | |
| 122 | +function getAnimationCssFromElement(ele) { | |
| 123 | + let animationrule = null; | |
| 124 | + let keyframename = null; | |
| 125 | + let keyframerule = null; | |
| 126 | + let outstr = ""; | |
| 127 | + | |
| 128 | + debuglog('アクションパレット情報(CSS) 取得開始 getAnimationCssFromElement() ***************'); | |
| 129 | + for (let i = 0; i < ele.sheet.cssRules.length; i++) { | |
| 130 | + animationrule = ele.sheet.cssRules[i]; | |
| 131 | + if (window.CSSRule.STYLE_RULE == animationrule.type) { | |
| 132 | + // animation定義文字列から keyframe nameを探して取得 | |
| 133 | + keyframename = getNameOfKeyframe(animationrule.style.cssText); | |
| 134 | + if (null != keyframename) { | |
| 135 | + // document内の全style sheetから引数のkeyframe定義を取得 | |
| 136 | + keyframerule = getKeyframeRule(keyframename); | |
| 137 | + } | |
| 138 | + // 指定keyframe定義がない or そもそもanimation定義でない or animation定義syntaxerr の場合は | |
| 139 | + // keyframe領域確保のためダミー定義を返す | |
| 140 | + if (null == keyframerule) { | |
| 141 | + console.error('animation ' + animationrule.style.cssText + 'に対応するkeyframe ' + keyframename + ' が見つからないため、ダミーを設定'); | |
| 142 | + keyframerule = '@keyframes dummy { to { color: black;} }'; | |
| 143 | + } | |
| 144 | + | |
| 145 | + // animation定義を復元 | |
| 146 | + let animationcss = animationrule.selectorText + ' {\r\n' + animationrule.style.cssText + '\r\n}'; | |
| 147 | + debuglog('パレット情報1件追加 ele.sheet.cssRules[' + i + ']\n' + animationcss + '\n' + keyframerule.cssText); | |
| 148 | + outstr += (animationcss + '\r\n' + keyframerule.cssText + '\r\n'); | |
| 149 | + } | |
| 150 | + } | |
| 151 | + debuglog('****************************************************'); | |
| 152 | + | |
| 153 | + return outstr; | |
| 154 | +}; | |
| 155 | +// 引数cssstrから animation-name(@keyframeで付けられる) を取得 | |
| 156 | +function getNameOfKeyframe( | |
| 157 | + cssstr // animation または animation-nameが定義された css文字列 | |
| 158 | +) { | |
| 159 | + // animation-nameでも shorthandでも keyframe nameはparamの先頭固定 | |
| 160 | + let name = null; | |
| 161 | + name = cssstr.match(/animation-name *: *(.+?) /); | |
| 162 | + if (null == name) { | |
| 163 | + name = cssstr.match(/animation *: *(.+?) /); | |
| 164 | + if (null == name) { | |
| 165 | + console.error('文字列から @keyframe名の検索に失敗 getNameOfKeyframe() str=' + cssstr); | |
| 166 | + return null; | |
| 167 | + } | |
| 168 | + } | |
| 169 | + return name[1]; | |
| 170 | +}; | |
| 171 | + | |
| 172 | +// 現在のdocumentに設定されている 全Style Sheetから | |
| 173 | +// ruleNameの名前がついた @keyframe検索 | |
| 174 | +// (Stack Overflowからのもらいもの) | |
| 175 | +function getKeyframeRule(ruleName) { | |
| 176 | + var ss = document.styleSheets, | |
| 177 | + rule, i, x; | |
| 178 | + | |
| 179 | + for (i = 0; i < ss.length; ++i) { | |
| 180 | + if (ss[i].cssRules) { | |
| 181 | + // loop through all the rules | |
| 182 | + | |
| 183 | + for (x = ss[i].cssRules.length - 1; x >= 0; x--) { | |
| 184 | + rule = ss[i].cssRules[x]; | |
| 185 | + if ((rule.type === window.CSSRule.KEYFRAMES_RULE || rule.type === window.CSSRule.WEBKIT_KEYFRAMES_RULE) && rule.name === ruleName) { | |
| 186 | + return rule; | |
| 187 | + } | |
| 188 | + } | |
| 189 | + } | |
| 190 | + } | |
| 191 | + return null; | |
| 192 | +} | |
| 193 | + | |
| 194 | + | |
| 195 | + | |
| 196 | + | |
| 197 | + | |
| 198 | + | |
| 199 | + | |
| 82 | 200 | // Drop fileから file objectを取得 (同期処理) |
| 83 | 201 | // 複数file 非対応。text or image がヒットした時点でcallbackを呼び出す |
| 84 | 202 | // callback書式(text,image): function(blob) |
| @@ -80,7 +80,6 @@ | ||
| 80 | 80 | // requestAnimationFrameのcallbackから呼び出される |
| 81 | 81 | render() { |
| 82 | 82 | while (0 < this.renderFunc.size()) { |
| 83 | - this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 84 | 83 | let renderFunction = this.renderFunc.dequeue(); |
| 85 | 84 | renderFunction(); |
| 86 | 85 | } |
| @@ -116,6 +115,7 @@ | ||
| 116 | 115 | // 画面に影響しない範囲で作成できるところまで Dom objectを作成する |
| 117 | 116 | let elType = (null != type.match(/imagebox/)) ? 'img' : 'div'; |
| 118 | 117 | this.DOMobject = document.createElement(elType); |
| 118 | + this.DOMobject.classList.add('DispObj'); | |
| 119 | 119 | this.DOMobject.dataset.objid = this.ObjID; |
| 120 | 120 | this.DOMobject.dataset.orgwidth = this.originalWidth; // transform: scaleの基準サイズ |
| 121 | 121 | this.DOMobject.dataset.orgheight = this.originalHeight; |
| @@ -185,13 +185,12 @@ | ||
| 185 | 185 | return new rectData(0, 0, destW, destH); |
| 186 | 186 | } |
| 187 | 187 | renderCreate() { |
| 188 | - this.DOMobject.style.position = "absolute"; // 決め打ち | |
| 188 | +// this.DOMobject.style.position = "absolute"; // DispObj CSS classで指定 | |
| 189 | 189 | this.DOMobject.style.left = this.x.toString() + "px"; |
| 190 | 190 | this.DOMobject.style.top = this.y.toString() + "px"; |
| 191 | 191 | this.DOMobject.style.width = this.width.toString() + "px"; |
| 192 | 192 | this.DOMobject.style.height = this.height.toString() + "px"; |
| 193 | 193 | this.DOMobject.style.border = "1px solid black"; // 指定するI/Fが必要 |
| 194 | -// this.DOMobject.style.backgroundColor = "#cdcdcd"; | |
| 195 | 194 | if (null != this.text) this.DOMobject.innerHTML = this.text; |
| 196 | 195 | |
| 197 | 196 | document.getElementById('DispField').appendChild(this.DOMobject); |
| @@ -203,6 +202,28 @@ | ||
| 203 | 202 | + ", border=" + this.DOMobject.style.border); |
| 204 | 203 | } |
| 205 | 204 | |
| 205 | + createFromElement(ele) { | |
| 206 | + this.debuglog('createFromElement() ele=' + ele); | |
| 207 | + | |
| 208 | + this.DOMobject = ele; | |
| 209 | + this.type = (null != ele.nodeName.match(/IMG/)) ? 'imagebox' : 'textbox'; | |
| 210 | + this.x = parseInt(ele.style.left); | |
| 211 | + this.y = parseInt(ele.style.top); | |
| 212 | + this.width = parseInt(ele.style.width); | |
| 213 | + this.height = parseInt(ele.style.height); | |
| 214 | + this.originalWidth = ele.dataset.orgwidth; | |
| 215 | + this.originalHeight = ele.dataset.orgheight; | |
| 216 | + this.ObjID = ele.dataset.objid; | |
| 217 | + this.degree = ele.dataset.degree; | |
| 218 | + this.DOMobject.addEventListener('mousedown', this.onMouseDown.bind(this), false); | |
| 219 | + | |
| 220 | + // 画像・テキストは既にelementに読み込み済み | |
| 221 | + // treeにも入っており、更新する必要もないのでこれで完了 | |
| 222 | + } | |
| 223 | + | |
| 224 | + | |
| 225 | + | |
| 226 | + | |
| 206 | 227 | // DIV box 移動 (移動先確定後の移動) |
| 207 | 228 | // ※ Drag中の表示は別で行っている |
| 208 | 229 | movebox(x, y) { |
| @@ -268,6 +289,8 @@ | ||
| 268 | 289 | } |
| 269 | 290 | // Mouse eventをappAreaからScalerに渡してもらうように設定 |
| 270 | 291 | window.appArea.setMouseEventObj(this.mouseMove.bind(this), this.mouseUp.bind(this)); |
| 292 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 293 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 271 | 294 | |
| 272 | 295 | |
| 273 | 296 | // Focus設定 |
| @@ -281,6 +304,9 @@ | ||
| 281 | 304 | |
| 282 | 305 | } |
| 283 | 306 | mouseMove(evt) { |
| 307 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 308 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 309 | + | |
| 284 | 310 | // 画像の仮移動 |
| 285 | 311 | this.endDragX = evt.pageX; |
| 286 | 312 | this.endDragY = evt.pageY; |
| @@ -302,6 +328,8 @@ | ||
| 302 | 328 | this.debuglog("mouseUp"); |
| 303 | 329 | // Mouse event処理対象をクリア |
| 304 | 330 | window.appArea.setMouseEventObj(null, null); |
| 331 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 332 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 305 | 333 | |
| 306 | 334 | |
| 307 | 335 | // 移動元として座標を保持 |
| @@ -39,7 +39,8 @@ | ||
| 39 | 39 | let styleaction = document.getElementById('stylepalact'); |
| 40 | 40 | let cssbg = stylebg.sheet.cssText; |
| 41 | 41 | let csstextcol = styletextcol.sheet.cssText; |
| 42 | - let cssaction = styleaction.sheet.cssText; | |
| 42 | + let cssaction = getAnimationCssFromElement(styleaction); | |
| 43 | + | |
| 43 | 44 | // 最終出力 |
| 44 | 45 | let savestr = '<style id="stylepalbg">' + cssbg + '</style><style id="stylepaltxtcol">' + csstextcol + '</style><style id="stylepalact">' + cssaction + '</style>' + displayInfo; |
| 45 | 46 |
| @@ -49,10 +50,11 @@ | ||
| 49 | 50 | this.DOMobject.innerHTML = text; |
| 50 | 51 | } |
| 51 | 52 | |
| 53 | + | |
| 54 | + | |
| 52 | 55 | // ブラウザ上への描画 ------------------------- |
| 53 | 56 | render() { |
| 54 | 57 | while (0 < this.renderFunc.size()) { |
| 55 | - // this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 56 | 58 | let renderFunction = this.renderFunc.dequeue(); |
| 57 | 59 | renderFunction(); |
| 58 | 60 | } |
| @@ -82,6 +82,29 @@ | ||
| 82 | 82 | } |
| 83 | 83 | |
| 84 | 84 | |
| 85 | + // 復元されたDOMに合わせて DispObjを作成する | |
| 86 | + // File Load 処理用 | |
| 87 | + adjustEnvToDOM() { | |
| 88 | + this.clear(); | |
| 89 | + | |
| 90 | + // 復元されたDispObj elementを取得 | |
| 91 | + let listEle = document.getElementsByClassName('DispObj'); | |
| 92 | + | |
| 93 | + this.debuglog('読み込み処理 DOM element設定から内部instanceを作成\nElement=' + listEle.length); | |
| 94 | + | |
| 95 | + let ele = null; | |
| 96 | + for (let i = 0; i < listEle.length; i++) { | |
| 97 | + ele = listEle[i]; | |
| 98 | + this.debuglog('読み込み処理 Element=' + (i+1) + '/' + listEle.length + '\ntype=' + ele.nodeName + ', objid=' + ele.dataset.objid); | |
| 99 | + let dispObj = new CDispObj(); | |
| 100 | + dispObj.createFromElement(ele); | |
| 101 | + | |
| 102 | + this.ObjIDLen = parseInt(ele.dataset.objid) + 1; | |
| 103 | + this.ObjIDarray[this.ObjIDLen - 1] = dispObj; | |
| 104 | + } | |
| 105 | + } | |
| 106 | + | |
| 107 | + | |
| 85 | 108 | // Command処理 ---------------------- |
| 86 | 109 | createDispObj(type, rect, opt) { |
| 87 | 110 | this.debuglog("createDispObj"); |
| @@ -33,7 +33,6 @@ | ||
| 33 | 33 | // requestAnimationFrameからの呼び出しを想定 |
| 34 | 34 | render() { |
| 35 | 35 | while (0 < this.renderFunc.size()) { |
| 36 | - // this.debuglog("render() renderFunc size=" + this.renderFunc.size()); | |
| 37 | 36 | let renderFunction = this.renderFunc.dequeue(); |
| 38 | 37 | renderFunction(); |
| 39 | 38 | } |
| @@ -28,7 +28,7 @@ | ||
| 28 | 28 | |
| 29 | 29 | // Mouse eventを処理するObj |
| 30 | 30 | setMouseEventObj(mouseMove, mouseUp) { |
| 31 | - this.debuglog('setMouseEventObj'); | |
| 31 | + this.debuglog('setMouseEventObj callbackMove=' + ((null != mouseMove)? mouseMove.name: 'null') + ', callbackUp=' + ((null != mouseUp)? mouseUp.name: 'null')); | |
| 32 | 32 | |
| 33 | 33 | this.mouseMoveFunc = mouseMove; |
| 34 | 34 | this.mouseUpFunc = mouseUp; |
| @@ -23,7 +23,7 @@ | ||
| 23 | 23 | window.postToWorker.setMessageHandler( receiveWorkerMessage ); |
| 24 | 24 | |
| 25 | 25 | |
| 26 | - // UI thread初期化 | |
| 26 | + // UI thread初期化 ---------------------- | |
| 27 | 27 | |
| 28 | 28 | // Mgr obj生成 |
| 29 | 29 | // Create Mgr--- |
| @@ -225,55 +225,47 @@ | ||
| 225 | 225 | |
| 226 | 226 | // 読み込み処理としては |
| 227 | 227 | // 1. 読み込みデータを style部分, html部分に分割 |
| 228 | - // 2. 取得したstyle部分からstyle要素を作成 | |
| 229 | - // 3. 既存のstyle要素と差し替え | |
| 230 | - // 4. html部分を DisplayFieldに読み込み | |
| 231 | - // 5. DOM treeに合わせて DispObj instanceを作成 | |
| 228 | + // 2. 既存のstyle要素と差し替え | |
| 229 | + // 3. html部分を DisplayFieldに読み込み | |
| 230 | + // 4. DOM treeに合わせて DispObj instanceを作成 | |
| 232 | 231 | |
| 233 | 232 | // 1. 読み込みデータを style部分, html部分に分割 |
| 234 | - let cssbg = str.match(/^<style id=\"stylepalbg\">([\s\S]*)<\/style>/)[1]; | |
| 235 | - let csstextcol = str.match(/<style id=\"stylepaltxtcol\">([\s\S]*)<\/style>/)[1]; | |
| 236 | - let cssaction = str.match(/<style id=\"stylepalact\">([\s\S]*)<\/style>/)[1]; | |
| 237 | - // 2. 取得したstyle部分からstyle要素を作成 | |
| 238 | - let newelebg = document.createElement('style'); | |
| 239 | - let neweletextcol = document.createElement('style'); | |
| 240 | - let neweleaction = document.createElement('style'); | |
| 241 | - newelebg.style.cssText = cssbg; | |
| 242 | - neweletextcol.style.cssText = csstextcol; | |
| 243 | - neweleaction.style.cssText = cssaction; | |
| 244 | - newelebg.setAttribute('id', 'stylepalbg'); | |
| 245 | - neweletextcol.setAttribute('id', 'stylepaltxtcol'); | |
| 246 | - neweleaction.setAttribute('id', 'stylepalact'); | |
| 233 | + // style部分取得 (palette group別) | |
| 234 | + let cssbg = str.match(/^<style id=\"stylepalbg\">([\s\S]*?)<\/style>/)[1]; | |
| 235 | + let csstextcol = str.match(/<style id=\"stylepaltxtcol\">([\s\S]*?)<\/style>/)[1]; | |
| 236 | + let cssaction = str.match(/<style id=\"stylepalact\">([\s\S]*?)<\/style>/)[1]; | |
| 237 | + // palette group別情報を さらにpalette要素別に分解 | |
| 238 | + let itemsbg = cssbg.match(/.palbg([\s\S]*?)\}/g); | |
| 239 | + let itemstextcol = csstextcol.match(/.paltxtcol([\s\S]*?)\}/g); | |
| 240 | + let itemsaction = cssaction.match(/.palact([\s\S]*?)\}/g); | |
| 241 | + let itemskeyframe = cssaction.match(/@keyframe([\s\S]*?)\}\s*\}/g); // @keyframeは中括弧が連続する '}}' 部分を終了判定にしている | |
| 247 | 242 | |
| 248 | - // 3. 既存のstyle要素と差し替え | |
| 249 | - // 現在設定されているstyle要素をDOM treeから外す | |
| 250 | - let oldelebg = document.getElementById('stylepalbg'); | |
| 251 | - let oldeletextcol = document.getElementById('stylepaltxtcol'); | |
| 252 | - let oldeleaction = document.getElementById('stylepalact'); | |
| 243 | + // 2. 既存のstyle要素と差し替え | |
| 244 | + let elebg = document.getElementById('stylepalbg'); | |
| 245 | + let eletextcol = document.getElementById('stylepaltxtcol'); | |
| 246 | + let eleaction = document.getElementById('stylepalact'); | |
| 253 | 247 | let eleapp = document.getElementById('apparea'); |
| 254 | -/* | |
| 255 | - eleapp.removeChild(oldelebg); | |
| 256 | - eleapp.removeChild(oldeletextcol); | |
| 257 | - eleapp.removeChild(oldeleaction); | |
| 248 | + for (let i = 0; i < 5; i++) { | |
| 249 | + setStrToCSS(elebg, i, itemsbg[i], null); | |
| 250 | + setStrToCSS(eletextcol, i, itemstextcol[i], null); | |
| 251 | + setStrToCSS(eleaction, i*2, itemsaction[i], null); | |
| 252 | + setStrToCSS(eleaction, i*2+1, itemskeyframe[i], null); | |
| 253 | + } | |
| 258 | 254 | |
| 259 | - // 読み込んだCSSで作成したstyle要素を追加 | |
| 260 | - let eleref = document.getElementsByClassName('palgroup')[0]; | |
| 261 | - eleapp.insertBefore(neweleaction, eleref); | |
| 262 | - eleapp.insertBefore(neweletextcol, neweleaction); | |
| 263 | - eleapp.insertBefore(newelebg, neweletextcol); | |
| 264 | -*/ | |
| 265 | - // 3. 既存のstyle要素と差し替え | |
| 266 | - eleapp.replaceChild(newelebg, oldelebg); | |
| 267 | - eleapp.replaceChild(neweletextcol, oldeletextcol); | |
| 268 | - eleapp.replaceChild(neweleaction, oldeleaction); | |
| 269 | - | |
| 270 | - // 4. html部分を DisplayFieldに読み込み | |
| 255 | + // 3. html部分を DisplayFieldに読み込み | |
| 271 | 256 | window.partsFocus.setClingingPartner(null); // Focus消去 |
| 272 | 257 | window.ObjIDMgr.clear(); // 表示中の要素を削除 (ObjIDMgr.deletebox()は隠すだけなので注意) |
| 273 | - let data = str.match(/<div style=[\s\S]*<\/div>/)[0]; | |
| 274 | - window.displayField.setFieldData(data); | |
| 258 | + // canvas上の要素は div or imgのみ。どちらが先頭かは不確定 | |
| 259 | + let data = str.match(/<(div|img) +class=[\s\S]*/); | |
| 260 | + if (null == data) { | |
| 261 | + console.error('File読み込み失敗 parseDdText() html要素がマッチしない\nindexDiv=' + indexDiv + ', indexImg=' + indexImg + ', str=' +str); | |
| 262 | + return; | |
| 263 | + } | |
| 264 | + // 表示データをそのまま DisplayFieldにセットする (見た目上は再現される) | |
| 265 | + window.displayField.setFieldData(data[0]); | |
| 275 | 266 | |
| 276 | - | |
| 267 | + // 4. DOM treeに合わせて DispObj instanceを作成 | |
| 268 | + window.ObjIDMgr.adjustEnvToDOM(); | |
| 277 | 269 | } |
| 278 | 270 | |
| 279 | 271 |
| @@ -280,6 +272,9 @@ | ||
| 280 | 272 | |
| 281 | 273 | |
| 282 | 274 | |
| 275 | + | |
| 276 | + | |
| 277 | + | |
| 283 | 278 | // 固定ボタン押下 event handler ---------------- |
| 284 | 279 | |
| 285 | 280 | function onClickPalButton(evt) { |
| @@ -65,7 +65,7 @@ | ||
| 65 | 65 | file // 画像を指しているfile object |
| 66 | 66 | ) { |
| 67 | 67 | let cssstr1 = '.palbg' + id + ' { background-image: url('; |
| 68 | - let cssstr2 = '); }'; | |
| 68 | + let cssstr2 = '); background-size: cover; }'; | |
| 69 | 69 | setImageFileToCSS(this.stylebg, id, cssstr1, cssstr2, file); |
| 70 | 70 | } |
| 71 | 71 |
| @@ -177,8 +177,11 @@ | ||
| 177 | 177 | // Mouse event |
| 178 | 178 | // 自elementをDragし、Drop時にTarget elementに自分のCSS classを適用する |
| 179 | 179 | onMouseDown(evt) { |
| 180 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 181 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 182 | + | |
| 180 | 183 | // Mouse eventをappAreaからCSSPalに渡してもらうように設定 |
| 181 | - window.appArea.setMouseEventObj(this.mouseMove.bind(this), null); | |
| 184 | + window.appArea.setMouseEventObj(this.mouseMove.bind(this), this.mouseUp.bind(this)); | |
| 182 | 185 | |
| 183 | 186 | // Drag対象を保持 |
| 184 | 187 | this.draggingDOM = evt.target; |
| @@ -191,6 +194,8 @@ | ||
| 191 | 194 | } |
| 192 | 195 | mouseMove(evt) { |
| 193 | 196 | if (null == this.draggingDOM) return; |
| 197 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 198 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 194 | 199 | |
| 195 | 200 | // 画像の仮移動 |
| 196 | 201 | this.endDragX = evt.pageX; |
| @@ -206,6 +211,8 @@ | ||
| 206 | 211 | //this.renderMoveDragging(moveX, moveY); |
| 207 | 212 | } |
| 208 | 213 | renderMoveDragging(x, y) { |
| 214 | + if (null == this.draggingDOM) return; | |
| 215 | + | |
| 209 | 216 | this.draggingDOM.style.left = x + "px"; |
| 210 | 217 | this.draggingDOM.style.top = y + "px"; |
| 211 | 218 | if( 1.0 == this.draggingDOM.style.opacity ) |
| @@ -212,6 +219,10 @@ | ||
| 212 | 219 | this.draggingDOM.style.opacity = 0.4; |
| 213 | 220 | } |
| 214 | 221 | mouseUp(evt) { |
| 222 | + if (null == this.draggingDOM) return; | |
| 223 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 224 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 225 | + | |
| 215 | 226 | let palobjid = this.draggingDOM.dataset.objid; |
| 216 | 227 | let palid = this.draggingDOM.dataset.palid; |
| 217 | 228 |
| @@ -56,6 +56,8 @@ | ||
| 56 | 56 | onMouseDown(evt) { |
| 57 | 57 | // Mouse eventをappAreaからCSSPalに渡してもらうように設定 |
| 58 | 58 | window.appArea.setMouseEventObj(this.mouseMove.bind(this), null); |
| 59 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 60 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 59 | 61 | |
| 60 | 62 | // Drag対象を保持 |
| 61 | 63 | this.draggingDOM = evt.target; |
| @@ -68,6 +70,8 @@ | ||
| 68 | 70 | } |
| 69 | 71 | mouseMove(evt) { |
| 70 | 72 | if (null == this.draggingDOM) return; |
| 73 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 74 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 71 | 75 | |
| 72 | 76 | // 画像の仮移動 |
| 73 | 77 | this.endDragX = evt.pageX; |
| @@ -90,6 +94,10 @@ | ||
| 90 | 94 | this.draggingDOM.style.top = y + "px"; |
| 91 | 95 | } |
| 92 | 96 | mouseUp(evt) { |
| 97 | + if (null == this.draggingDOM) return; | |
| 98 | + evt.stopPropagation(); // event伝播を自分のところで止める(上位に行かなくなる) | |
| 99 | + evt.preventDefault(); // 要素既定のdefault動作を止める | |
| 100 | + | |
| 93 | 101 | // Drag中 element情報をクリア |
| 94 | 102 | this.draggingDOM = null; |
| 95 | 103 | // Mouse event callback設定をクリア |
| @@ -1,59 +0,0 @@ | ||
| 1 | -<!DOCTYPE html> | |
| 2 | -<html lang="ja"> | |
| 3 | -<head> | |
| 4 | - <meta charset="UTF-8"> | |
| 5 | - <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> | |
| 6 | - <meta name="viewport" content="user-scalable=no, initial-scale=1, maximum-scale=1.5, minimum-scale=0.1, width=device-width, height=device-height"> | |
| 7 | - <title></title> | |
| 8 | - </head> | |
| 9 | -<body> | |
| 10 | - | |
| 11 | - <style id="stylepalbg">.palbg0 { | |
| 12 | - background-image: url(""); | |
| 13 | -} | |
| 14 | -.palbg1 { | |
| 15 | - background-image: url("dummy.png"); | |
| 16 | -} | |
| 17 | -.palbg2 { | |
| 18 | - background-image: url(""); | |
| 19 | -} | |
| 20 | -.palbg3 { | |
| 21 | - background-image: url("dummy.png"); | |
| 22 | -} | |
| 23 | -.palbg4 { | |
| 24 | - background-image: url("dummy.png"); | |
| 25 | -} | |
| 26 | -</style><style id="stylepaltxtcol">.paltxtcol0 { | |
| 27 | - color: black; | |
| 28 | -} | |
| 29 | -.paltxtcol1 { | |
| 30 | - color: red; | |
| 31 | -} | |
| 32 | -.paltxtcol2 { | |
| 33 | - color: blue; | |
| 34 | -} | |
| 35 | -.paltxtcol3 { | |
| 36 | - color: green; | |
| 37 | -} | |
| 38 | -.paltxtcol4 { | |
| 39 | - color: brown; | |
| 40 | -} | |
| 41 | -</style><style id="stylepalact">.palact0 { | |
| 42 | - animation:animeY1 0.5s ease-in 0.3s; | |
| 43 | -} | |
| 44 | -.palact1 { | |
| 45 | - animation:animeact1 2s ease-in 0.3s; | |
| 46 | -} | |
| 47 | -.palact2 { | |
| 48 | - animation:animeact2 2s ease-in 0.3s; | |
| 49 | -} | |
| 50 | -.palact3 { | |
| 51 | - animation:animeact3 2s ease-in 0.3s; | |
| 52 | -} | |
| 53 | -.palact4 { | |
| 54 | - animation:animeact4 2s ease-in 0.3s; | |
| 55 | -} | |
| 56 | -</style> | |
| 57 | - <div style="border: 1px solid black; border-image: none; left: 150px; top: 150px; width: 50px; height: 50px; position: absolute; opacity: 1;" data-objid="0" data-orgwidth="50" data-orgheight="50" data-degree="0"></div><div class="palbg0" style="border: 1px solid black; border-image: none; left: 42px; top: 65px; width: 83px; height: 111px; font-size: 222%; position: absolute; opacity: 1; transform: rotateZ(9.37209deg);" data-objid="1" data-focus="true" data-orgwidth="50" data-orgheight="50" data-degree="9.372092708579089"></div><div class="palbg2" style="border: 1px solid black; border-image: none; left: 196px; top: 104px; width: 130px; height: 117px; font-size: 234%; position: absolute; opacity: 1; transform: rotateZ(-15.3661deg);" data-objid="2" data-focus="true" data-orgwidth="50" data-orgheight="50" data-degree="-15.366085212823378">テキスト | |
| 58 | -Text</div> | |
| 59 | -</body> | |
| \ No newline at end of file |