Compare commits

..

7 Commits

Author SHA1 Message Date
guofei 1418128b0d fix: update some styles 2024-10-17 13:58:36 +08:00
guofei a6196e8f7d fix: img bg prohibit drag 2024-10-17 11:07:36 +08:00
icefire 133e6ca036 update: 样式调整 2024-10-16 18:39:55 +08:00
icefire 2e34a926a1 update: 样式调整 2024-10-16 18:31:10 +08:00
icefire 93fb34532f feat: 手机框替换 2024-10-15 21:13:17 +08:00
icefire d79e186bb8 feat: 修改select下拉图标 2024-10-14 22:56:39 +08:00
icefire 8b1faf408d feat: 首页ui改版 2024-10-14 22:38:47 +08:00
27 changed files with 230 additions and 125 deletions

View File

@ -48,24 +48,25 @@
} }
.pane-btn { .pane-btn {
border-radius: 4px; border-radius: 8px;
border: 0; border: 3px solid #494949;
display: inline-block; display: inline-block;
height: 30px; height: 62px;
line-height: 30px; width: 62px;
width: 150px;
cursor: pointer; cursor: pointer;
outline: none; outline: none;
font-weight: bold; font-weight: bold;
background: #494949;
color: #eee;
box-sizing: border-box;
} }
.pane-btn.checked { .pane-btn.checked {
background-color: #02b564; border: 3px solid #00DE7A;
color: white;
} }
.pane-btn + .pane-btn { .pane-btn + .pane-btn {
margin-left: 10px; margin-left: 14px;
} }
.generate-btn { .generate-btn {
@ -85,17 +86,16 @@
display: flex; display: flex;
align-items: center; align-items: center;
position: fixed; position: fixed;
right: 40px; right: 24px;
top: 30px; top: 24px;
z-index: 1000; z-index: 1000;
} }
.language-select-container .language-select { .language-select-container .language-select {
margin-left: 10px; width: 156px;
width: 135px; height: 42px;
height: 35px; background-color: #36383A;
background-color: #eef1f6; color: #EEEEEE;
color: #333333;
font-size: 14px; font-size: 14px;
display: flex; display: flex;
align-items: center; align-items: center;
@ -103,12 +103,16 @@
.language-select-warpper { .language-select-warpper {
position: relative; position: relative;
border-radius: 5px;
background: #36383A;
border: 1px solid #5B5B5B;
box-shadow: 0px 3px 0px 0px rgba(0, 0, 0, 0.3);
} }
.language-select-warpper .language-icon { .language-select-warpper .language-icon {
position: absolute; position: absolute;
left: 18px; left: 14px;
top: 8px; top: 11px;
z-index: 2000; z-index: 2000;
} }

View File

@ -28,6 +28,17 @@ import queryString from "query-string";
import { withTranslation } from "react-i18next"; import { withTranslation } from "react-i18next";
import i18n from "i18next"; import i18n from "i18next";
import languageImg from "./images/language.png"; import languageImg from "./images/language.png";
import Arrow from './images/arrow.svg'
import SvgIcon from '@mui/material/SvgIcon';
function CustomIcon(props) {
return (
<SvgIcon viewBox="0 0 24 24" {...props}>
<path d="M7.41 7.84L12 12.42l4.59-4.58L18 9.25l-6 6-6-6z" fill="white" />
</SvgIcon>
);
}
class App extends React.Component { class App extends React.Component {
constructor(props) { constructor(props) {
@ -285,6 +296,7 @@ class App extends React.Component {
className="language-select" className="language-select"
value={language} value={language}
label={t("language")} label={t("language")}
IconComponent={CustomIcon}
onChange={(e) => { onChange={(e) => {
let lang = e.target.value; let lang = e.target.value;
this.setState({ this.setState({

View File

@ -1,33 +1,45 @@
.deviceFrameContainer { .deviceFrameContainer {
width: 100%; width: 100%;
height: 100vh;
flex: 1 1 0%; flex: 1 1 0%;
align-items: center; align-items: center;
position: relative; position: relative;
display: flex; display: flex;
height: 100vh;
justify-content: center; justify-content: center;
overflow: hidden; overflow: hidden;
background: #36383A;
} }
.refreshButton { .refreshButton {
position: absolute; position: absolute;
left: 30px; right: 24px;
bottom: 30px; bottom: 24px;
cursor: pointer; cursor: pointer;
} }
.qrCode { .qrCode {
position: absolute; position: absolute;
right: 40px; right: 24px;
top: 100px; top: 82px;
background: #DBDBDB;
width: 156px;
text-align: center;
border-radius: 5px;
padding-top: 14px;
padding-bottom: 11px;
} }
.qrCode p { .qrCode p {
display: flex; display: flex;
justify-content: center;
align-items: center; align-items: center;
padding-left: 23px; color: #666666;
font-size: 14px;
} }
.qrCode img { .qrCode img {
margin-left: 6px; }
#deviceFrame {
border-radius: 10px;
} }

View File

@ -3,7 +3,7 @@ import androidHPng from "../../images/android-h.png";
import androidVPng from "../../images/android-v.png"; import androidVPng from "../../images/android-v.png";
import iphoneHPng from "../../images/iphone-h.png"; import iphoneHPng from "../../images/iphone-h.png";
import iphoneVPng from "../../images/iphone-v.png"; import iphoneVPng from "../../images/iphone-v.png";
import ipadHPng from "../../images/iPad-h.png"; import ipadHPng from "../../images/ipad-horizontal.png";
import ipadVPng from "../../images/iPad-v.png"; import ipadVPng from "../../images/iPad-v.png";
import copyPng from "../../images/copy.png"; import copyPng from "../../images/copy.png";
import refreshButtonPng from "../../images/refresh-button.png"; import refreshButtonPng from "../../images/refresh-button.png";
@ -12,41 +12,63 @@ import React, { useEffect, useState } from "react";
import { BaseUrl } from "../../constants"; import { BaseUrl } from "../../constants";
import { Alert, Snackbar } from "@mui/material"; import { Alert, Snackbar } from "@mui/material";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import zIndex from "@mui/material/styles/zIndex";
var configMap = { var configMap = {
"android-h": { "android-h": {
bkg: androidHPng, bkg: androidVPng,
width: "512px", rotate: true,
height: "286px", width: "666px",
offsetv: "5px", height: "321px",
offsetv: "4px",
}, },
"android-v": { "android-v": {
bkg: androidVPng, bkg: androidVPng,
width: "286px", width: "321px",
height: "512px", height: "666px",
offsetv: "5px", offsetv: "5px",
}, },
"iphone-h": { "iphone-h": {
bkg: iphoneHPng, bkg: iphoneVPng,
width: "628px", rotate: true,
height: "322px", width: "690px",
height: "325px",
styles: {
borderRadius: '13px'
}
}, },
"iphone-v": { "iphone-v": {
bkg: iphoneVPng, bkg: iphoneVPng,
width: "322px", width: "322px",
height: "628px", height: "693px",
styles: {
borderRadius: '13px'
}
}, },
"ipad-h": { "ipad-h": {
bkg: ipadHPng, bkg: ipadHPng,
width: "433px", rotate: true,
width: "458px",
height: "326px", height: "326px",
offsetv: "5px", offsetv: "5px",
imgStyle: {
transform: 'rotate(-90deg) scale(1.4)',
},
styles: {
transform: "translate(-50%, -50%) scale(1.42)",
}
}, },
"ipad-v": { "ipad-v": {
bkg: ipadVPng, bkg: ipadHPng,
width: "324px", width: "326px",
height: "433px", height: "458px",
offsetv: "5px", offsetv: "5px",
imgStyle: {
transform: 'scale(1.4)',
},
styles: {
transform: "translate(-50%, -50%) scale(1.42)",
}
}, },
}; };
@ -75,13 +97,14 @@ function DeviceFrame(props) {
<img src={refreshButtonPng} alt="refresh" onClick={() => refresh()} className="refreshButton" /> <img src={refreshButtonPng} alt="refresh" onClick={() => refresh()} className="refreshButton" />
{props.qrDataUrl && ( {props.qrDataUrl && (
<div className="qrCode"> <div className="qrCode">
<p style={{ margin: "0" }}> <img src={props.qrDataUrl} alt="qr code" />
<p style={{ margin: "0", }}>
{t("scan")} {t("scan")}
<CopyToClipboard text={absoluteUrl} onCopy={() => setOpenPopup(true)}> <CopyToClipboard text={absoluteUrl} onCopy={() => setOpenPopup(true)}>
<img style={{ cursor: "pointer", width: "20px", height: "20px" }} src={copyPng} alt="复制链接" /> <img style={{ cursor: "pointer", width: "20px", height: "20px" }} src={copyPng} alt="复制链接" />
</CopyToClipboard> </CopyToClipboard>
</p> </p>
<img src={props.qrDataUrl} alt="qr code" />
</div> </div>
)} )}
<Snackbar <Snackbar
@ -97,7 +120,7 @@ function DeviceFrame(props) {
</Alert> </Alert>
</Snackbar> </Snackbar>
<div> <div>
<img src={config.bkg} alt="background" /> <img src={config.bkg} alt="background" style={{ position: "relative", zIndex: 100, transform: config.rotate ? "rotate(-90deg)" : "", pointerEvents: 'none', ...config.imgStyle }} />
{props.htmlUrl && ( {props.htmlUrl && (
<iframe <iframe
id="deviceFrame" id="deviceFrame"
@ -105,8 +128,9 @@ function DeviceFrame(props) {
position: "absolute", position: "absolute",
top: "50%", top: "50%",
left: "50%", left: "50%",
transform: "translate(-50%, -50%)", transform: "translate(-50%, -50%) scale(1.007)",
paddingBottom: config.offsetv, paddingBottom: config.offsetv,
...config.styles
}} }}
allow="clipboard-read; clipboard-write *" allow="clipboard-read; clipboard-write *"
title="preview" title="preview"

View File

@ -1,43 +1,58 @@
.Pane { .Pane {
width: 350px; width: 370px;
background-color: rgb(52, 90, 176); background-color: #535353;
min-height: 100vh; min-height: 100vh;
display: flex; padding: 24px;
flex-direction: column; box-sizing: border-box;
display: flex;
flex-direction: column;
position: relative;
} }
.modeSelect { .modeSelect {
width: 100%; width: 100%;
border: none; border: none;
} }
.deviceIcon { .deviceIcon {
width: auto; width: auto;
height: 80px; height: 76px;
margin-right: 10px; margin-bottom: 10px;
margin-bottom: -24px; margin-top: 7px;
}
.deviceText {
} }
.deviceContent { .deviceContent {
text-align: left; height: 100%;
color: white; color: #eeeeee;
} }
.deviceButtons { .deviceButtons {
width: 160px; height: 100%;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
} }
.deviceButtons img { .deviceButtons img {
cursor: pointer; cursor: pointer;
} }
.selectPager { .selectPager {
left: 0 !important; left: 25px !important;
width: 350px; width: 319px !important;
} }
.bottomLogo { .bottomLogo {
margin-top: auto; position: absolute;
} width: 322px;
bottom: 24px;
text-align: center;
img {
width: 280px;
}
}
.Mui-focused .MuiOutlinedInput-notchedOutline {
border: none;
}

View File

@ -4,8 +4,11 @@ import MenuItem from "@mui/material/MenuItem";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { CircularProgress, List, ListItem } from "@mui/material"; import { CircularProgress, List, ListItem } from "@mui/material";
import ipadPng from "../../images/ipad.png"; import ipadPng from "../../images/ipad.png";
import ipadTextPng from "../../images/ipad-text.png";
import iphonePng from "../../images/iphone.png"; import iphonePng from "../../images/iphone.png";
import iphoneTextPng from '../../images/iphone-text.png'
import androidPng from "../../images/android.png"; import androidPng from "../../images/android.png";
import androidTextPng from '../../images/android-text.png'
import logoPanePng from "../../images/logo-pane.png"; import logoPanePng from "../../images/logo-pane.png";
import horizentalChecked from "../../images/horizental-button-checked.png"; import horizentalChecked from "../../images/horizental-button-checked.png";
import horizentalButton from "../../images/horizental-button.png"; import horizentalButton from "../../images/horizental-button.png";
@ -13,23 +16,35 @@ import verticalChecked from "../../images/vertical-button-checked.png";
import verticalButton from "../../images/vertical-button.png"; import verticalButton from "../../images/vertical-button.png";
import modeSelectPng from "../../images/mode-select.png"; import modeSelectPng from "../../images/mode-select.png";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import SvgIcon from '@mui/material/SvgIcon';
function CustomIcon(props) {
return (
<SvgIcon viewBox="0 0 24 24" {...props}>
<path d="M7.41 7.84L12 12.42l4.59-4.58L18 9.25l-6 6-6-6z" fill="white" />
</SvgIcon>
);
}
var deviceConfigs = [ var deviceConfigs = [
{ {
name: "iPhone", name: "iPhone",
icon: iphonePng, icon: iphonePng,
text: iphoneTextPng,
vertical: "iphone-v", vertical: "iphone-v",
horizental: "iphone-h", horizental: "iphone-h",
}, },
{ {
name: "Android Phone", name: "Android Phone",
icon: androidPng, icon: androidPng,
text: androidTextPng,
vertical: "android-v", vertical: "android-v",
horizental: "android-h", horizental: "android-h",
}, },
{ {
name: "iPad", name: "iPad",
icon: ipadPng, icon: ipadPng,
text: ipadTextPng,
vertical: "ipad-v", vertical: "ipad-v",
horizental: "ipad-h", horizental: "ipad-h",
}, },
@ -64,39 +79,77 @@ function Pane(props) {
return ( return (
<div className="Pane"> <div className="Pane">
<Select <div>
id="mode-select" {props.loading ? (
value={mode} <CircularProgress color="inherit" />
onChange={(e) => modeChange(e)} ) : (
startAdornment={<img src={modeSelectPng} height={"30px"} alt="selected" />} project && (
<Select
id="project-select"
value={project}
onChange={(e) => projectSelect(e)}
IconComponent={CustomIcon}
placeholder="选择项目"
sx={{
width: "100%",
height: '42px',
border: "1px solid #757575",
background: "#535353",
borderRadius: '5px',
color: "white",
boxShadow: '0px 3px 0px 0px rgba(0, 0, 0, 0.3)',
// '&.MuiOutlinedInput-notchedOutline': {
// borderColor: '#757575', // 设置焦点时的边框颜色为红色
// },
'&.Mui-focused': {
borderColor: '#757575',
'&.MuiOutlinedInput-notchedOutline': {
borderColor: '#757575 !important',
},
},
}}
MenuProps={{
classes: {
paper: "selectPager",
},
}}
>
{props.projects.map((pName, i) => (
<MenuItem key={i} value={pName}>
{pName}
</MenuItem>
))}
</Select>
)
)}
</div>
<List
sx={{ sx={{
width: "100%", paddingTop: '20px'
border: "none",
background: "rgb(69, 115, 191)",
color: "white",
fontWeight: "bold",
}} }}
MenuProps={{ >
classes: {
paper: "selectPager",
},
}}>
<MenuItem value={"normal"}>Normal Preview</MenuItem>
<MenuItem value={"dynamic"}>Dynamic Preview</MenuItem>
</Select>
<List>
{deviceConfigs.map((_device) => ( {deviceConfigs.map((_device) => (
<ListItem <ListItem
key={_device.name} key={_device.name}
sx={{ sx={{
display: "flex", display: "flex",
textAlign: "center", textAlign: "center",
justifyContent: "space-between",
borderRadius: '5px',
background: '#535353',
border: '1px solid #757575',
boxShadow: '0px 3px 0px 0px rgba(0, 0, 0, 0.3)',
paddingLeft: '23px',
paddingRight: '20px',
marginBottom: '14px'
}}> }}>
<div style={{ width: "120px", alignItems: "center" }}> <div style={{ display: "flex", flexDirection: 'column', justifyContent: 'center', alignItems: "center", width: '98px' }}>
<img className="deviceIcon" src={_device.icon} alt={_device.name} /> <img className="deviceIcon" style={{ height: _device.name === 'iPad' ? '54px' : '76px' }} src={_device.icon} alt={_device.name} />
<img className="deviceText" src={_device.text} alt={_device.name} />
</div> </div>
<div className="deviceContent"> <div className="deviceContent">
<p style={{ margin: "14px 0" }}>{_device.name}</p>
<div className="deviceButtons"> <div className="deviceButtons">
<button className={`pane-btn ${device === _device.horizental ? "checked" : ""}`} onClick={() => deviceChange(_device.horizental)}> <button className={`pane-btn ${device === _device.horizental ? "checked" : ""}`} onClick={() => deviceChange(_device.horizental)}>
{t("horizontal")} {t("horizontal")}
@ -111,35 +164,8 @@ function Pane(props) {
</ListItem> </ListItem>
))} ))}
</List> </List>
<div
style={{ {!props.hideLogo && <div className="bottomLogo"><img src={logoPanePng} alt="logo" width={"100%"} /></div>}
marginTop: "50px",
color: "white",
}}>
{props.loading ? (
<CircularProgress color="inherit" />
) : (
project && (
<Select
id="project-select"
value={project}
onChange={(e) => projectSelect(e)}
placeholder="选择项目"
sx={{
background: "rgb(115, 158, 211)",
width: "80%",
color: "white",
}}>
{props.projects.map((pName, i) => (
<MenuItem key={i} value={pName}>
{pName}
</MenuItem>
))}
</Select>
)
)}
</div>
{!props.hideLogo && <img src={logoPanePng} alt="logo" width={"100%"} className="bottomLogo" />}
</div> </div>
); );
} }

View File

@ -1,4 +1,5 @@
var BaseUrl = "https://api.soyootech.com/"; // var BaseUrl = "https://api.soyootech.com/";
var BaseUrl = "api/";
export async function UserLogin(loginForm) { export async function UserLogin(loginForm) {
try { try {
var response = await fetch(BaseUrl + "SoyooUser/previewlogin", { var response = await fetch(BaseUrl + "SoyooUser/previewlogin", {

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 KiB

After

Width:  |  Height:  |  Size: 3.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.7 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 15 KiB

After

Width:  |  Height:  |  Size: 4.9 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.4 KiB

After

Width:  |  Height:  |  Size: 912 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 201 B

View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" fill="none" version="1.1" width="13.435028612613678" height="13.435028612613678" viewBox="0 0 13.435028612613678 13.435028612613678"><g transform="matrix(0.7071067690849304,-0.7071067690849304,0.7071067690849304,0.7071067690849304,-4.335786265016024,2.9675144346638707)"><path d="M0.41421353816986084,15.217514306306839L8.91421353816986,15.217514306306839L8.91421353816986,13.217514306306839L2.414213538169861,13.217514306306839L2.414213538169861,6.717514306306839L0.41421353816986084,6.717514306306839L0.41421353816986084,15.217514306306839Z" fill-rule="evenodd" fill="#D8D8D8" fill-opacity="1"/></g></svg>

After

Width:  |  Height:  |  Size: 690 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 3.2 KiB

After

Width:  |  Height:  |  Size: 386 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 18 KiB

After

Width:  |  Height:  |  Size: 2.9 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.8 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 671 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.7 KiB

After

Width:  |  Height:  |  Size: 474 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 918 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 22 KiB

After

Width:  |  Height:  |  Size: 5.8 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.7 KiB

After

Width:  |  Height:  |  Size: 821 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 6.2 KiB

After

Width:  |  Height:  |  Size: 778 B

Binary file not shown.

Before

Width:  |  Height:  |  Size: 26 KiB

After

Width:  |  Height:  |  Size: 9.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.5 KiB

After

Width:  |  Height:  |  Size: 1.9 KiB

View File

@ -0,0 +1,10 @@
import SvgIcon from '@mui/material/SvgIcon';
import React from "react";
export function ArrowIcon(props) {
return (
<SvgIcon viewBox="0 0 24 24" {...props}>
<path d="M7.41 7.84L12 12.42l4.59-4.58L18 9.25l-6 6-6-6z" fill="white" />
</SvgIcon>
);
}

View File

@ -6,7 +6,7 @@
"submit": "Submit", "submit": "Submit",
"noAuth": "No permission", "noAuth": "No permission",
"loginError": "Account or password is incorrect", "loginError": "Account or password is incorrect",
"horizontal": "Horizontal", "horizontal": "Hor",
"vertical": "Vertical", "vertical": "Vertical",
"scan": "Mobile preview", "scan": "Mobile preview",
"none": "None", "none": "None",

View File

@ -1,12 +1,12 @@
const { createProxyMiddleware } = require("http-proxy-middleware"); const { createProxyMiddleware } = require("http-proxy-middleware");
module.exports = function (app) { module.exports = function (app) {
// app.use( app.use(
// "/api", "/api",
// createProxyMiddleware({ createProxyMiddleware({
// target: "https://api.soyootech.com/", target: "https://api.soyootech.com/",
// changeOrigin: true, changeOrigin: true,
// pathRewrite: { "^/api": "" }, pathRewrite: { "^/api": "" },
// }) })
// ); );
}; };