3 Commits

Author SHA1 Message Date
9e67513897 build(npm): update gulp-haxetool 2026-05-20 00:05:30 +03:00
e41d461ba6 feat(app): update nginx image source 2026-05-20 00:03:52 +03:00
acf0706bef docs: add screenshot 2026-05-04 16:12:46 +03:00
15 changed files with 13793 additions and 13665 deletions

View File

@@ -1,9 +1,13 @@
# Puzzlez # Puzzle'z
Puzzle game
![Puzzle'z](docs/Screenshot_2026-05-04_15-42-42.jpg "Puzzle'z")
## Play ## Play
https://shmyga.ru/puzzlez/html5/index.html https://shmyga.ru/puzzlez/html5/index.html
## Packages ## Releases
https://git.shmyga.ru/InfernalGames/-/packages/generic/puzzlez https://git.shmyga.ru/InfernalGames/puzzlez/releases

Binary file not shown.

After

Width:  |  Height:  |  Size: 357 KiB

View File

@@ -4,8 +4,11 @@ const Config = require("./config.json");
const packageInfo = require("./package.json"); const packageInfo = require("./package.json");
const { System, Sdk, Haxe, Project } = require("gulp-haxetool"); const { System, Sdk, Haxe, Project } = require("gulp-haxetool");
const dateformat = require("dateformat"); const dateformat = require("dateformat");
const argv = require("yargs").argv;
const publish = require("./tasks/gulp-publish"); const publish = require("./tasks/gulp-publish");
Project.useRuffle();
if (packageInfo.haxe) { if (packageInfo.haxe) {
Haxe.VERSION = packageInfo.haxe; Haxe.VERSION = packageInfo.haxe;
} }
@@ -63,6 +66,9 @@ const config = new Project.Config({
flags: ["proto_debug"], flags: ["proto_debug"],
}); });
const host = argv.host || "localhost";
const port = argv.port || 5000;
const app = new Project( const app = new Project(
Project.BuildSystem.OPENFL, Project.BuildSystem.OPENFL,
[ [
@@ -87,6 +93,10 @@ const app = new Project(
width: 1280, width: 1280,
height: 768, height: 768,
}, },
macros: [
`CompilationOption.set('host','${host}')`,
`CompilationOption.set('port',${port})`,
],
flags: ["app"], flags: ["app"],
}), }),
).bind(module, gulp); ).bind(module, gulp);

27325
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -9,7 +9,7 @@
"gulp-add": "0.0.2", "gulp-add": "0.0.2",
"gulp-clean": "^0.4.0", "gulp-clean": "^0.4.0",
"gulp-cli": "^2.2.0", "gulp-cli": "^2.2.0",
"gulp-haxetool": "^0.1.9", "gulp-haxetool": "^0.2.1",
"yargs": "^13.2.4" "yargs": "^13.2.4"
}, },
"haxeDependencies": { "haxeDependencies": {
@@ -20,8 +20,7 @@
"svg": "1.1.3", "svg": "1.1.3",
"protohx": "0.4.6", "protohx": "0.4.6",
"yield": "3.2.2", "yield": "3.2.2",
"formatter": "1.16.0", "formatter": "1.16.0"
"hxWebSockets": "1.4.0"
}, },
"haxe": "4.2.5" "haxe": "4.2.5"
} }

View File

@@ -4,7 +4,6 @@ import hw.app.App;
import hw.app.Const; import hw.app.Const;
import hw.log.TraceLogger; import hw.log.TraceLogger;
import ru.m.puzzlez.image.ImageSourceBundle; import ru.m.puzzlez.image.ImageSourceBundle;
import ru.m.puzzlez.net.Network;
import ru.m.puzzlez.render.part.IPartBuilder; import ru.m.puzzlez.render.part.IPartBuilder;
import ru.m.puzzlez.settings.Settings; import ru.m.puzzlez.settings.Settings;
import ru.m.puzzlez.source.AssetImageSource; import ru.m.puzzlez.source.AssetImageSource;
@@ -19,14 +18,12 @@ import ru.m.update.Updater;
class PuzzlezApp { class PuzzlezApp {
@:provide static var updater:Updater; @:provide static var updater:Updater;
@:provide static var sourceBundle:ImageSourceBundle; @:provide static var sourceBundle:ImageSourceBundle;
@:provide static var network:Network;
public static function main() { public static function main() {
// ToDo: fix @:provide macro // ToDo: fix @:provide macro
Settings; Settings;
IPartBuilder; IPartBuilder;
GameStorage; GameStorage;
Network;
sourceBundle.register(new AssetImageSource()); sourceBundle.register(new AssetImageSource());
sourceBundle.register(new FileImageSource()); sourceBundle.register(new FileImageSource());
sourceBundle.register(new PixabayImageSource(CompilationOption.get("PIXABAY_KEY"))); sourceBundle.register(new PixabayImageSource(CompilationOption.get("PIXABAY_KEY")));
@@ -39,6 +36,5 @@ class PuzzlezApp {
app.icon = openfl.Assets.getBitmapData("resources/icon.png"); app.icon = openfl.Assets.getBitmapData("resources/icon.png");
app.view = new PuzzlezAppView(); app.view = new PuzzlezAppView();
L.d("Puzzlez", "started"); L.d("Puzzlez", "started");
network.auth();
} }
} }

View File

@@ -4,9 +4,8 @@ import hw.connect.ConnectionFactory;
import hw.connect.IConnection; import hw.connect.IConnection;
import hw.signal.Signal; import hw.signal.Signal;
import hw.storage.SharedObjectStorage; import hw.storage.SharedObjectStorage;
import promhx.Deferred;
import promhx.Promise; import promhx.Promise;
import ru.m.data.DataSource; import ru.m.data.IDataSource;
import ru.m.puzzlez.proto.core.User; import ru.m.puzzlez.proto.core.User;
import ru.m.puzzlez.proto.event.GameAction; import ru.m.puzzlez.proto.event.GameAction;
import ru.m.puzzlez.proto.event.GameEvent; import ru.m.puzzlez.proto.event.GameEvent;
@@ -14,7 +13,6 @@ import ru.m.puzzlez.proto.game.GamePreset;
import ru.m.puzzlez.proto.game.GameState; import ru.m.puzzlez.proto.game.GameState;
import ru.m.puzzlez.proto.pack.AuthRequest; import ru.m.puzzlez.proto.pack.AuthRequest;
import ru.m.puzzlez.proto.pack.GameActionRequest; import ru.m.puzzlez.proto.pack.GameActionRequest;
import ru.m.puzzlez.proto.pack.GameCreateRequest;
import ru.m.puzzlez.proto.pack.GameJoinRequest; import ru.m.puzzlez.proto.pack.GameJoinRequest;
import ru.m.puzzlez.proto.pack.GameLeaveRequest; import ru.m.puzzlez.proto.pack.GameLeaveRequest;
import ru.m.puzzlez.proto.pack.GameListRequest; import ru.m.puzzlez.proto.pack.GameListRequest;
@@ -23,20 +21,7 @@ import ru.m.puzzlez.proto.pack.NotificationResponse;
import ru.m.puzzlez.proto.pack.Request; import ru.m.puzzlez.proto.pack.Request;
import ru.m.puzzlez.proto.pack.Response; import ru.m.puzzlez.proto.pack.Response;
class SignalUtil { @:provide class Network implements IDataSource<String, GameState> {
public static function next<T>(signal:Signal<T>):Promise<T> {
var d:Deferred<T> = new Deferred<T>();
var receiver:T->Void;
receiver = (value:T) -> {
signal.disconnect(receiver);
d.resolve(value);
};
signal.connect(receiver);
return d.promise();
}
}
@:provide class Network implements DataSource<GameState> {
public var userSignal:Signal<User> = new Signal(); public var userSignal:Signal<User> = new Signal();
public var notificationSignal:Signal<NotificationResponse> = new Signal(); public var notificationSignal:Signal<NotificationResponse> = new Signal();
public var listSignal:Signal<GameListResponse> = new Signal(); public var listSignal:Signal<GameListResponse> = new Signal();
@@ -66,17 +51,17 @@ class SignalUtil {
public function auth():Promise<User> { public function auth():Promise<User> {
connection.send(new Request().setAuth(new AuthRequest().setUser(restoreUser()))); connection.send(new Request().setAuth(new AuthRequest().setUser(restoreUser())));
return SignalUtil.next(userSignal); return userSignal.next();
} }
public function createGame(preset:GamePreset):Promise<GameState> { public function createGame(preset:GamePreset):Promise<GameState> {
connection.send(new Request().setCreate(new GameCreateRequest().setPreset(preset))); connection.send(new Request().setJoin(new GameJoinRequest().setPreset(preset)));
return SignalUtil.next(joinSignal); return joinSignal.next();
} }
public function joinGame(gameId:String):Promise<GameState> { public function joinGame(gameId:String):Promise<GameState> {
connection.send(new Request().setJoin(new GameJoinRequest().setGameId(gameId))); connection.send(new Request().setJoin(new GameJoinRequest().setGameId(gameId)));
return SignalUtil.next(joinSignal); return joinSignal.next();
} }
public function leaveGame():Void { public function leaveGame():Void {
@@ -117,7 +102,7 @@ class SignalUtil {
public function getPage(page:Page):Promise<DataPage<GameState>> { public function getPage(page:Page):Promise<DataPage<GameState>> {
connection.send(new Request().setList(new GameListRequest().setCount(page.count).setPage(page.index))); connection.send(new Request().setList(new GameListRequest().setCount(page.count).setPage(page.index)));
return SignalUtil.next(listSignal).then((list:GameListResponse) -> ({ return listSignal.next().then((list:GameListResponse) -> ({
page: { page: {
index: list.page, index: list.page,
count: list.count, count: list.count,
@@ -128,4 +113,8 @@ class SignalUtil {
data: list.games, data: list.games,
})); }));
} }
public function get(id:String):GameState {
return null;
}
} }

View File

@@ -2,7 +2,7 @@ package ru.m.puzzlez.net;
import ru.m.puzzlez.proto.event.GameStart; import ru.m.puzzlez.proto.event.GameStart;
import hw.signal.Signal; import hw.signal.Signal;
import ru.m.puzzlez.game.IGame; import ru.m.puzzlez.image.IGame;
import ru.m.puzzlez.proto.event.GameAction; import ru.m.puzzlez.proto.event.GameAction;
import ru.m.puzzlez.proto.event.GameEvent; import ru.m.puzzlez.proto.event.GameEvent;
import ru.m.puzzlez.proto.game.GameState; import ru.m.puzzlez.proto.game.GameState;

View File

@@ -25,6 +25,7 @@ class NginxImageSource implements ImageSource {
public var id(default, never):String = "nginx"; public var id(default, never):String = "nginx";
private var baseUrl:String; private var baseUrl:String;
private var cache:Map<String, Promise<NginxResponse>> = new Map();
public function new(baseUrl:String) { public function new(baseUrl:String) {
this.baseUrl = baseUrl; this.baseUrl = baseUrl;
@@ -36,31 +37,35 @@ class NginxImageSource implements ImageSource {
public function getPage(page:Page):Promise<DataPage<ImageId>> { public function getPage(page:Page):Promise<DataPage<ImageId>> {
var category = page.filter.get("category"); var category = page.filter.get("category");
return new JsonLoader<NginxResponse>().GET(category != null ? this.baseUrl + category + "/" : this.baseUrl) var url = category != null ? this.baseUrl + category + "/" : this.baseUrl;
.then((response:NginxResponse) -> { if (!this.cache.exists(url)) {
var data:Array<ImageId> = []; this.cache.set(url, new JsonLoader<NginxResponse>().GET(url));
for (item in response) { }
if (item.type == NginxResponseItemType.FILE) { return this.cache.get(url).then((response:NginxResponse) -> {
var itemId = category != null ? category + "@" + item.name : item.name; var data:Array<ImageId> = [];
data.push(new ImageId().setSource(id).setId(itemId)); for (item in response) {
} if (item.type == NginxResponseItemType.FILE) {
var itemId = category != null ? category + "@" + item.name : item.name;
data.push(new ImageId().setSource(id).setId(itemId));
} }
return { }
page: page, data = data.slice(page.index * page.count, (page.index + 1) * page.count);
data: data, return {
total: response.length, page: page,
} data: data,
}); total: response.length,
}
});
} }
public function load(id:String, thumb:Bool = false):Promise<ImageValue> { public function load(id:String, thumb:Bool = false):Promise<ImageValue> {
var url = this.baseUrl + StringTools.replace(id, "@", "/"); var url = this.baseUrl + StringTools.replace(id, "@", "/");
if (thumb) { // TODO: default size by screen width?
var parts = url.split("."); var width = thumb ? 360 : 1920;
var index = parts.length - 2; var parts = url.split(".");
parts[index] = parts[index] + "-thumbnail"; var index = parts.length - 2;
url = parts.join("."); parts[index] = parts[index] + '-w${width}';
} url = parts.join(".");
return Promise.promise(ImageValue.URL(url)); return Promise.promise(ImageValue.URL(url));
} }

View File

@@ -7,7 +7,6 @@ import hw.view.popup.ConfirmView;
import promhx.Promise; import promhx.Promise;
import ru.m.puzzlez.game.Game; import ru.m.puzzlez.game.Game;
import ru.m.puzzlez.game.IGame; import ru.m.puzzlez.game.IGame;
import ru.m.puzzlez.net.NetworkGame;
import ru.m.puzzlez.proto.event.GameAction; import ru.m.puzzlez.proto.event.GameAction;
import ru.m.puzzlez.proto.event.GameEvent; import ru.m.puzzlez.proto.event.GameEvent;
import ru.m.puzzlez.proto.event.gameaction.Action; import ru.m.puzzlez.proto.event.gameaction.Action;
@@ -38,7 +37,7 @@ import ru.m.puzzlez.view.popup.PreviewPopup;
L.d("Frame", '$ID: ${state.preset.image.source}:${state.preset.image.id}'); L.d("Frame", '$ID: ${state.preset.image.source}:${state.preset.image.id}');
onHide(); onHide();
if (state.online) { if (state.online) {
game = new NetworkGame(state); // game = new NetworkGame(state);
} else { } else {
game = new Game(state); game = new Game(state);
} }

View File

@@ -6,7 +6,6 @@ import hw.view.form.ToggleButtonView;
import hw.view.frame.FrameSwitcher; import hw.view.frame.FrameSwitcher;
import hw.view.frame.FrameView; import hw.view.frame.FrameView;
import ru.m.puzzlez.game.GameUtil; import ru.m.puzzlez.game.GameUtil;
import ru.m.puzzlez.net.Network;
import ru.m.puzzlez.proto.game.ImageId; import ru.m.puzzlez.proto.game.ImageId;
import ru.m.puzzlez.view.common.PresetView; import ru.m.puzzlez.view.common.PresetView;
@@ -18,7 +17,7 @@ import ru.m.puzzlez.view.common.PresetView;
@:provide var switcher:FrameSwitcher; @:provide var switcher:FrameSwitcher;
@:provide var network:Network; // @:provide var network:Network;
private var imageId:ImageId; private var imageId:ImageId;
public function new() { public function new() {
@@ -56,7 +55,7 @@ import ru.m.puzzlez.view.common.PresetView;
private function start(online:Bool = false):Void { private function start(online:Bool = false):Void {
if (online) { if (online) {
network.createGame(imageView.state.preset).then(state -> switcher.change(GameFrame.ID, state)); // network.createGame(imageView.state.preset).then(state -> switcher.change(GameFrame.ID, state));
} else { } else {
switcher.change(GameFrame.ID, imageView.state); switcher.change(GameFrame.ID, imageView.state);
} }

View File

@@ -29,7 +29,7 @@ views:
geometry.margin.left: 15 geometry.margin.left: 15
text: Network text: Network
+onPress: ~start(true) +onPress: ~start(true)
visible: true visible: false
- id: image - id: image
$type: ru.m.puzzlez.view.common.PresetView $type: ru.m.puzzlez.view.common.PresetView
geometry.stretch: true geometry.stretch: true

View File

@@ -1,6 +1,5 @@
package ru.m.puzzlez.view; package ru.m.puzzlez.view;
import ru.m.puzzlez.net.Network;
import hw.view.data.DataView; import hw.view.data.DataView;
import hw.view.form.ButtonView; import hw.view.form.ButtonView;
import hw.view.frame.FrameSwitcher; import hw.view.frame.FrameSwitcher;
@@ -29,7 +28,6 @@ import ru.m.update.Updater;
@:provide var fileStorage:FileStorage; @:provide var fileStorage:FileStorage;
@:provide var gameStorage:GameStorage; @:provide var gameStorage:GameStorage;
@:provide var sourceBundle:ImageSourceBundle; @:provide var sourceBundle:ImageSourceBundle;
@:provide var network:Network;
private var fileSource:ImageListConfig = {title: "Files", sourceId: "file"}; private var fileSource:ImageListConfig = {title: "Files", sourceId: "file"};
private var startedGames:GameListConfig = { private var startedGames:GameListConfig = {
@@ -42,10 +40,6 @@ import ru.m.update.Updater;
source: gameStorage, source: gameStorage,
filter: ["status" => GameStatus.COMPLETE] filter: ["status" => GameStatus.COMPLETE]
}; };
private var networkGames:GameListConfig = {
title: "Network",
source: network
};
public function new() { public function new() {
super(ID); super(ID);

View File

@@ -39,10 +39,6 @@ views:
$type: hw.view.form.ButtonView $type: hw.view.form.ButtonView
text: Completed text: Completed
+onPress: ~showGames(completedGames) +onPress: ~showGames(completedGames)
- id: networkButton
$type: hw.view.form.ButtonView
text: Network
+onPress: ~showGames(networkGames)
- $type: hw.view.form.ButtonView - $type: hw.view.form.ButtonView
text: Upload text: Upload
geometry.margin.left: 30 geometry.margin.left: 30

View File

@@ -5,8 +5,8 @@ import ru.m.puzzlez.proto.pack.GameListResponse;
import ru.m.puzzlez.proto.pack.GameListRequest; import ru.m.puzzlez.proto.pack.GameListRequest;
import hw.connect.session.ProtoSession; import hw.connect.session.ProtoSession;
import hw.log.BaseLogger.LoggerUtil; import hw.log.BaseLogger.LoggerUtil;
import ru.m.puzzlez.game.Game; import ru.m.puzzlez.image.Game;
import ru.m.puzzlez.game.GameUtil; import ru.m.puzzlez.image.GameUtil;
import ru.m.puzzlez.proto.core.User; import ru.m.puzzlez.proto.core.User;
import ru.m.puzzlez.proto.event.GameAction; import ru.m.puzzlez.proto.event.GameAction;
import ru.m.puzzlez.proto.event.GameEvent; import ru.m.puzzlez.proto.event.GameEvent;