index.ts 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. import { Effect, ModuleControl } from "queenjs";
  2. import { EditorModule } from "../../module";
  3. import { reactive } from "vue";
  4. import { DesignComp } from "../../objects/DesignTemp/DesignComp";
  5. import { RxValue } from "../ReactCtrl/rxValue";
  6. import { CardState } from "../../objects/DesignTemp";
  7. import { isPc } from "@queenjs/utils";
  8. const PCConst = {
  9. minRate: 0.42,
  10. maxRate: 0.8,
  11. height: 1520,
  12. factor: 0.8067,
  13. widths: [1900, 2356, 2920],
  14. ranges: [[0.64537, 0.8], [0.52063, 0.64537], [0.42, 0.52063]]
  15. }
  16. const MobileConst = {
  17. minRate: 0.4495,
  18. maxRate: 0.75,
  19. width: 750,
  20. factor: 0.84312,
  21. heights: [1000, 1186, 1406],
  22. ranges: [[0.63234, 0.75], [0.5331387120830735, 0.632340125298328], [0.4495, 0.5331387120830735]]
  23. }
  24. const AllSize = ["short","normal", "long"];
  25. export const MobleScreenNames = ["短屏", "普通", "长屏幕"];
  26. export class ScreenCtrl extends ModuleControl<EditorModule> {
  27. state = RxValue.create({
  28. screen: {
  29. useFor: "mobile" as "mobile" | "pc", // pc 还是 手机
  30. pageMode: "long" as "long" | "short", //长页面还是短页面(一屏高度)
  31. pageSizeType: "normal" as "normal" | "long" | "short", //适配类型 长屏 正常 短屏幕
  32. },
  33. docWidth: window.innerWidth, //实际屏幕宽度
  34. docHeight: window.innerHeight, //实际屏幕高度
  35. safeFactor: 0.8,
  36. currScreenId: ""
  37. })
  38. get isShortPage () {
  39. return this.state.screen.pageMode == "short";
  40. }
  41. get isLongPage() {
  42. return this.state.screen.pageMode == "long";
  43. }
  44. get currScreenId() {
  45. return this.state.screen.useFor + this.state.screen.pageMode + this.state.screen.pageSizeType;
  46. }
  47. initEvent() {
  48. window.addEventListener("resize", ()=>{
  49. if ( !this.store.isDisplay ) return;
  50. console.log("apply screen");
  51. this.applyScreen();
  52. })
  53. if ( !this.store.isEditMode ) return;
  54. this.state.onScreenChanged(()=>this.updateAdapterState());
  55. const {factor, sizes, ranges} = this.createSteps(MobileConst.width, MobileConst.minRate, MobileConst.maxRate)
  56. console.log("range=>", ranges, factor, sizes);
  57. console.log( this.getRightScreenId() );
  58. }
  59. createSteps(base: number, min:number, max:number, size = 3) {
  60. const factor = Math.pow((min / max) , 1.0 / size);
  61. const ranges = [];
  62. const sizes = [];
  63. for (let i=0; i<size; i++) {
  64. const w = factor * base / min
  65. let pre = min;
  66. min = base / w;
  67. sizes.push( w )
  68. ranges.push([pre, min])
  69. }
  70. const t = sizes[0]
  71. sizes[0] = sizes[2]
  72. sizes[2] = t;
  73. const t2 = ranges[0]
  74. ranges[0] = ranges[2]
  75. ranges[2] = t2;
  76. return {factor, sizes, ranges}
  77. }
  78. getAdapterIndex() {
  79. if(this.state.screen.pageMode == "long") return 1;
  80. return ["short", "normal", "long"].indexOf(this.state.screen.pageSizeType)
  81. }
  82. applyScreen() {
  83. const option = this.getRightScreenId();
  84. if ( this.isScreenExit(option.sreenId) ) {
  85. this.restorScreenPage(option.sreenId);
  86. this.state.currScreenId = option.sreenId;
  87. return;
  88. }
  89. let n= AllSize.length;
  90. while(n--) {
  91. const screenId = option.device + option.pageMode + AllSize[n];
  92. if ( this.isScreenExit(screenId) ) {
  93. this.restorScreenPage(screenId);
  94. this.state.currScreenId = screenId;
  95. return;
  96. }
  97. }
  98. if (option.pageMode == "short") {
  99. option.pageMode = "long";
  100. return;
  101. }
  102. if (option.pageMode == "long") {//如果长页面模式找一个合适端页面拼成长页面
  103. let n= AllSize.length;
  104. while(n--) {
  105. const screenId = option.device + "short" + AllSize[n];
  106. if ( this.isScreenExit(screenId) ) {
  107. this.restorScreenPage(screenId);
  108. this.state.currScreenId = screenId;
  109. return;
  110. }
  111. }
  112. }
  113. }
  114. getRightScreenId() {
  115. const currScreen = isPc() ? "pc" : "mobile";
  116. const rate = isPc() ? window.innerHeight / window.innerWidth : window.innerWidth / window.innerHeight;
  117. const sizes = currScreen == "pc" ? PCConst.ranges : MobileConst.ranges;
  118. let n = 3;
  119. let index = 1;
  120. while(n--) {
  121. const m = sizes[n]
  122. if (rate >= m[0]) {
  123. index = n;
  124. }
  125. }
  126. if (this.state.screen.pageMode == "long") index = 1;
  127. console.log("rate=>", rate);
  128. const sreenId = currScreen + this.state.screen.pageMode + ["short", "normal", "long"][index];
  129. return {sreenId, index, device: currScreen, pageMode: this.state.screen.pageMode};
  130. }
  131. updateAdapterState() {
  132. const page = this.controls.pageCtrl;
  133. if (!page.rootPage) return;
  134. this.restorScreenPage();
  135. let total = 0;
  136. page.streamCardIds.forEach(c=>{
  137. const card = this.helper.findComp(c) as DesignComp;
  138. card.setW(this.getCurrScreenWidth());
  139. total += this.helper.extendStreamCard(card.id);
  140. })
  141. const w = this.helper.designSizeToPx(this.getCurrScreenWidth());
  142. let h = this.helper.designSizeToPx(this.getCurrScreenHeight());
  143. if (this.state.screen.pageMode == "long") {
  144. h = this.helper.designSizeToPx(total)
  145. }
  146. this.controls.editorCtrl.state.setPage({w, h});
  147. page.rootPage.layout.size[0] = this.getCurrScreenWidth();
  148. page.rootPage.layout.size[1] = this.getCurrScreenHeight();
  149. this.state.safeFactor = this.state.screen.useFor == "pc" ? PCConst.factor : MobileConst.factor;
  150. this.state.currScreenId = this.currScreenId;
  151. }
  152. isScreenExit(screenId:string) {
  153. const page = this.controls.pageCtrl;
  154. return !!page.designData.compScreenMap[screenId] &&
  155. page.designData.compScreenMap[screenId].length > 0 &&
  156. page.designData.compScreenMap[screenId][0].children.length > 0;
  157. }
  158. restorScreenPage(screenId = "") {
  159. const page = this.controls.pageCtrl;
  160. if (!page.rootPage) return;
  161. //获取当前screen的配置
  162. screenId = screenId ? screenId : this.state.screen.useFor + this.state.screen.pageMode + this.state.screen.pageSizeType;
  163. const screenCards = page.designData.compScreenMap[screenId] || [];
  164. console.log("apply screen=>", screenId);
  165. //刷新当前card的配置
  166. page.streamCardIds.forEach(c=>{
  167. const card = this.helper.findComp(c) as DesignComp;
  168. const screenCard = screenCards.find(item=>item.id == c)
  169. let newChilds:string[] = [];
  170. let childrs = screenCard?.children || [];
  171. if (screenCard && screenCard.size){
  172. card.layout.size[0] = screenCard.size[0];
  173. card.layout.size[1] = screenCard.size[1];
  174. }
  175. childrs.forEach(item=>{
  176. newChilds.push(item.id);
  177. const screenComp = this.helper.findComp(item.id) as DesignComp;
  178. screenComp.layout.size[0] = item.size[0];
  179. screenComp.layout.size[1] = item.size[1];
  180. screenComp.layout.transformMatrix = item.matrix;
  181. })
  182. card.children.default = newChilds;
  183. })
  184. }
  185. saveScreenPage() {
  186. const page = this.controls.pageCtrl;
  187. //获取当前screen的配置
  188. const screenId = this.state.screen.useFor + this.state.screen.pageMode + this.state.screen.pageSizeType;
  189. const screenCards = ScreenCtrl.createScreenCards(this.store.compMap, page.rootPage);
  190. if (!page.designData.compScreenMap) page.designData.compScreenMap = {};
  191. page.designData.compScreenMap[screenId] = screenCards;
  192. }
  193. static createScreenCards(compMap:any, rootPage: DesignComp) {
  194. //获取当前screen的配置
  195. const screenCards = [] as CardState [];
  196. const streamCardIds = rootPage.children.default || [];
  197. streamCardIds.forEach(c=>{
  198. const card = compMap[c] as DesignComp;
  199. const screenCard :CardState = {
  200. id: c,
  201. size: [card.layout.size[0], card.layout.size[1]],
  202. children: []
  203. }
  204. let childrs = card.children.default || [];
  205. childrs.forEach(item=>{
  206. const c = compMap[item] as DesignComp;
  207. screenCard.children.push({id: item, size: c.layout.size, matrix: c.layout.transformMatrix as string} );
  208. })
  209. screenCards.push(screenCard);
  210. })
  211. return screenCards;
  212. }
  213. getCurrScreenHeight() {
  214. const pageValue = this.state
  215. if ( pageValue.screen.useFor == "pc") {
  216. return PCConst.height;
  217. }
  218. const currScreenIndex = this.getAdapterIndex();
  219. return MobileConst.heights[currScreenIndex];
  220. }
  221. getCurrScreenWidth() {
  222. const currScreenIndex = this.getAdapterIndex();
  223. const pageValue = this.state
  224. if ( pageValue.screen.useFor == "pc" ) {
  225. return PCConst.widths[currScreenIndex];
  226. }
  227. return MobileConst.width;
  228. }
  229. }