♻️ Brand new post list

This commit is contained in:
LittleSheep 2024-03-02 20:01:59 +08:00
parent 178f80c707
commit 3ae72cd9e0
23 changed files with 314 additions and 65 deletions

View File

@ -5,10 +5,8 @@ import "time"
type Post struct { type Post struct {
BaseModel BaseModel
Alias string `json:"alias" gorm:"uniqueIndex"`
Title string `json:"title"`
Content string `json:"content"` Content string `json:"content"`
Tags []Tag `json:"tags" gorm:"many2many:post_tags"` Hashtags []Tag `json:"tags" gorm:"many2many:post_tags"`
Categories []Category `json:"categories" gorm:"many2many:post_categories"` Categories []Category `json:"categories" gorm:"many2many:post_categories"`
Attachments []Attachment `json:"attachments"` Attachments []Attachment `json:"attachments"`
LikedAccounts []PostLike `json:"liked_accounts"` LikedAccounts []PostLike `json:"liked_accounts"`

View File

@ -12,12 +12,12 @@ import (
func getOwnPost(c *fiber.Ctx) error { func getOwnPost(c *fiber.Ctx) error {
user := c.Locals("principal").(models.Account) user := c.Locals("principal").(models.Account)
id := c.Params("postId") id, _ := c.ParamsInt("postId", 0)
take := c.QueryInt("take", 0) take := c.QueryInt("take", 0)
offset := c.QueryInt("offset", 0) offset := c.QueryInt("offset", 0)
tx := database.C.Where(&models.Post{ tx := database.C.Where(&models.Post{
Alias: id, BaseModel: models.BaseModel{ID: uint(id)},
AuthorID: user.ID, AuthorID: user.ID,
}) })

View File

@ -13,12 +13,12 @@ import (
) )
func getPost(c *fiber.Ctx) error { func getPost(c *fiber.Ctx) error {
id := c.Params("postId") id, _ := c.ParamsInt("postId", 0)
take := c.QueryInt("take", 0) take := c.QueryInt("take", 0)
offset := c.QueryInt("offset", 0) offset := c.QueryInt("offset", 0)
tx := database.C.Where(&models.Post{ tx := database.C.Where(&models.Post{
Alias: id, BaseModel: models.BaseModel{ID: uint(id)},
}).Where("published_at <= ? OR published_at IS NULL", time.Now()) }).Where("published_at <= ? OR published_at IS NULL", time.Now())
post, err := services.GetPost(tx) post, err := services.GetPost(tx)
@ -162,8 +162,6 @@ func createPost(c *fiber.Ctx) error {
post, err := services.NewPost( post, err := services.NewPost(
user, user,
realm, realm,
data.Alias,
data.Title,
data.Content, data.Content,
data.Attachments, data.Attachments,
data.Categories, data.Categories,
@ -207,8 +205,6 @@ func editPost(c *fiber.Ctx) error {
post, err := services.EditPost( post, err := services.EditPost(
post, post,
data.Alias,
data.Title,
data.Content, data.Content,
data.PublishedAt, data.PublishedAt,
data.Categories, data.Categories,

View File

@ -5,7 +5,7 @@ import (
"strings" "strings"
"time" "time"
"code.smartsheep.studio/hydrogen/identity/pkg/views" "code.smartsheep.studio/hydrogen/interactive/pkg/views"
"github.com/gofiber/fiber/v2" "github.com/gofiber/fiber/v2"
"github.com/gofiber/fiber/v2/middleware/cache" "github.com/gofiber/fiber/v2/middleware/cache"
"github.com/gofiber/fiber/v2/middleware/cors" "github.com/gofiber/fiber/v2/middleware/cors"

View File

@ -20,7 +20,7 @@ func PreloadRelatedPost(tx *gorm.DB) *gorm.DB {
Preload("Author"). Preload("Author").
Preload("Attachments"). Preload("Attachments").
Preload("Categories"). Preload("Categories").
Preload("Tags"). Preload("Hashtags").
Preload("RepostTo"). Preload("RepostTo").
Preload("ReplyTo"). Preload("ReplyTo").
Preload("RepostTo.Author"). Preload("RepostTo.Author").
@ -29,8 +29,8 @@ func PreloadRelatedPost(tx *gorm.DB) *gorm.DB {
Preload("ReplyTo.Attachments"). Preload("ReplyTo.Attachments").
Preload("RepostTo.Categories"). Preload("RepostTo.Categories").
Preload("ReplyTo.Categories"). Preload("ReplyTo.Categories").
Preload("RepostTo.Tags"). Preload("RepostTo.Hashtags").
Preload("ReplyTo.Tags") Preload("ReplyTo.Hashtags")
} }
func FilterPostWithCategory(tx *gorm.DB, alias string) *gorm.DB { func FilterPostWithCategory(tx *gorm.DB, alias string) *gorm.DB {
@ -161,7 +161,7 @@ WHERE t.id IN ?`, prefix, prefix, prefix, prefix, prefix), postIds).Scan(&reactI
func NewPost( func NewPost(
user models.Account, user models.Account,
realm *models.Realm, realm *models.Realm,
alias, title, content string, content string,
attachments []models.Attachment, attachments []models.Attachment,
categories []models.Category, categories []models.Category,
tags []models.Tag, tags []models.Tag,
@ -202,11 +202,9 @@ func NewPost(
} }
post = models.Post{ post = models.Post{
Alias: alias,
Title: title,
Content: content, Content: content,
Attachments: attachments, Attachments: attachments,
Tags: tags, Hashtags: tags,
Categories: categories, Categories: categories,
AuthorID: user.ID, AuthorID: user.ID,
RealmID: realmId, RealmID: realmId,
@ -225,7 +223,7 @@ func NewPost(
BaseModel: models.BaseModel{ID: *post.ReplyID}, BaseModel: models.BaseModel{ID: *post.ReplyID},
}).Preload("Author").First(&op).Error; err == nil { }).Preload("Author").First(&op).Error; err == nil {
if op.Author.ID != user.ID { if op.Author.ID != user.ID {
postUrl := fmt.Sprintf("https://%s/posts/%s", viper.GetString("domain"), post.Alias) postUrl := fmt.Sprintf("https://%s/posts/%d", viper.GetString("domain"), post.ID)
err := NotifyAccount( err := NotifyAccount(
op.Author, op.Author,
fmt.Sprintf("%s replied you", user.Name), fmt.Sprintf("%s replied you", user.Name),
@ -252,7 +250,7 @@ func NewPost(
}) })
for _, account := range accounts { for _, account := range accounts {
postUrl := fmt.Sprintf("https://%s/posts/%s", viper.GetString("domain"), post.Alias) postUrl := fmt.Sprintf("https://%s/posts/%d", viper.GetString("domain"), post.ID)
err := NotifyAccount( err := NotifyAccount(
account, account,
fmt.Sprintf("%s just posted a post", user.Name), fmt.Sprintf("%s just posted a post", user.Name),
@ -270,7 +268,7 @@ func NewPost(
func EditPost( func EditPost(
post models.Post, post models.Post,
alias, title, content string, content string,
publishedAt *time.Time, publishedAt *time.Time,
categories []models.Category, categories []models.Category,
tags []models.Tag, tags []models.Tag,
@ -294,11 +292,9 @@ func EditPost(
publishedAt = lo.ToPtr(time.Now()) publishedAt = lo.ToPtr(time.Now())
} }
post.Alias = alias
post.Title = title
post.Content = content post.Content = content
post.PublishedAt = *publishedAt post.PublishedAt = *publishedAt
post.Tags = tags post.Hashtags = tags
post.Categories = categories post.Categories = categories
post.Attachments = attachments post.Attachments = attachments

Binary file not shown.

6
pkg/views/embed.go Normal file
View File

@ -0,0 +1,6 @@
package views
import "embed"
//go:embed all:dist
var FS embed.FS

View File

@ -2,7 +2,7 @@
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8"> <meta charset="UTF-8">
<link rel="icon" href="/favicon.ico"> <link rel="icon" href="/favicon.svg">
<meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Goatplaza</title> <title>Goatplaza</title>
</head> </head>

View File

@ -13,9 +13,11 @@
"format": "prettier --write src/" "format": "prettier --write src/"
}, },
"dependencies": { "dependencies": {
"@fontsource/roboto": "^5.0.8",
"@mdi/font": "^7.4.47", "@mdi/font": "^7.4.47",
"@unocss/reset": "^0.58.5", "@unocss/reset": "^0.58.5",
"pinia": "^2.1.7", "pinia": "^2.1.7",
"universal-cookie": "^7.1.0",
"unocss": "^0.58.5", "unocss": "^0.58.5",
"vue": "^3.4.15", "vue": "^3.4.15",
"vue-router": "^4.2.5", "vue-router": "^4.2.5",

21
pkg/views/public/favicon.svg Executable file
View File

@ -0,0 +1,21 @@
<svg version="1.2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1024 1024" width="1024" height="1024">
<title>SmartSheep Logo</title>
<defs>
<image width="124" height="198" id="img1" href=""/>
<image width="122" height="142" id="img2" href=""/>
</defs>
<style>
.s0 { fill: #ffffff;stroke: #000000;stroke-miterlimit:100;stroke-width: 56 }
.s1 { fill: #4750a3;stroke: #000000;stroke-miterlimit:100;stroke-width: 56 }
</style>
<path id="Wool" fill-rule="evenodd" class="s0" d="m128 608.4c0 95.9 77.4 173.6 172.8 173.6h441.6c84.8 0 153.6-69.1 153.6-154.3 0-74.6-52.8-136.9-122.9-151.1 4.9-12.9 7.7-27 7.7-41.7 0-63.9-51.6-115.8-115.2-115.8-23.6 0-45.7 7.3-64 19.6-33.2-57.9-95.2-96.7-166.4-96.7-106.1 0-192 86.3-192 192.9 0 3.2 0.1 6.5 0.2 9.7-67.2 23.8-115.4 88.1-115.4 163.8z"/>
<g id="Crystal">
<path id="Crystal" class="s1" d="m699 224l138.6 80v160l-138.6 80-138.6-80v-160z"/>
<use id="Highlight" href="#img1" x="688" y="255"/>
</g>
<g id="Horn">
</g>
<g id="Face">
<use id="Slime" href="#img2" x="233" y="538"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 8.1 KiB

View File

@ -0,0 +1,11 @@
html, body, #app, .v-application {
overflow: auto !important;
}
.no-scrollbar {
scrollbar-width: none;
}
.no-scrollbar::-webkit-scrollbar {
width: 0;
}

View File

@ -0,0 +1,31 @@
<template>
<v-card>
<template #text>
<div class="flex gap-3">
<div>
<v-avatar
color="grey-lighten-2"
icon="mdi-account-circle"
class="rounded-card"
:src="props.item?.author.avatar"
/>
</div>
<div>
<div class="font-bold">{{ props.item?.author.nick }}</div>
{{ props.item?.content }}
</div>
</div>
</template>
</v-card>
</template>
<script setup lang="ts">
const props = defineProps<{ item: any }>();
</script>
<style scoped>
.rounded-card {
border-radius: 8px;
}
</style>

View File

@ -0,0 +1,21 @@
<template>
<div class="post-list">
<div v-if="props.loading" class="text-center py-8">
<v-progress-circular indeterminate />
</div>
<v-infinite-scroll :items="props.posts" :onLoad="props.loader">
<template v-for="(item, index) in props.posts" :key="item">
<div class="mb-3 px-1">
<post-item :item="item" />
</div>
</template>
</v-infinite-scroll>
</div>
</template>
<script setup lang="ts">
import PostItem from "@/components/posts/PostItem.vue";
const props = defineProps<{ loading: boolean, posts: any[], loader: (opts: any) => Promise<any> }>();
</script>

View File

@ -1,13 +1,24 @@
<template> <template>
<v-navigation-drawer v-model="drawerOpen" color="grey-lighten-5" floating> <v-navigation-drawer v-model="drawerOpen" color="grey-lighten-5" floating>
<div class="d-flex text-center justify-center items-center h-[64px]"> <v-list density="compact" nav>
<h1>Goatplaza</h1> </v-list>
</div>
</v-navigation-drawer> </v-navigation-drawer>
<v-app-bar height="64" color="primary" scroll-behavior="elevate" flat> <v-app-bar height="64" color="primary" scroll-behavior="elevate" flat>
<div class="container mx-auto px-5"> <div class="max-md:px-5 md:px-12 flex flex-grow-1 items-center">
<v-app-bar-nav-icon variant="text" @click.stop="toggleDrawer"></v-app-bar-nav-icon> <v-app-bar-nav-icon variant="text" @click.stop="toggleDrawer" />
<router-link :to="{ name: 'explore' }">
<h2 class="ml-2 text-lg font-500">Goatplaza</h2>
</router-link>
<v-spacer />
<v-tooltip v-for="item in navigationMenu" :text="item.name" location="bottom">
<template #activator="{ props }">
<v-btn flat v-bind="props" :to="{ name: item.to }" size="small" :icon="item.icon" />
</template>
</v-tooltip>
</div> </div>
</v-app-bar> </v-app-bar>
@ -16,12 +27,16 @@
</v-main> </v-main>
</template> </template>
<script setup> <script setup lang="ts">
import { ref } from "vue" import { ref } from "vue";
const drawerOpen = ref(true) const navigationMenu = [
{ name: "Explore", icon: "mdi-compass", to: "explore" }
];
const drawerOpen = ref(true);
function toggleDrawer() { function toggleDrawer() {
drawerOpen.value = !drawerOpen.value drawerOpen.value = !drawerOpen.value;
} }
</script> </script>

View File

@ -1,27 +1,35 @@
import "virtual:uno.css" import "virtual:uno.css";
import { createApp } from "vue" import "./assets/utils.css";
import { createPinia } from "pinia"
import "vuetify/styles" import { createApp } from "vue";
import { createVuetify } from "vuetify" import { createPinia } from "pinia";
import * as components from "vuetify/components"
import * as directives from "vuetify/directives"
import "@mdi/font/css/materialdesignicons.min.css" import "vuetify/styles";
import { createVuetify } from "vuetify";
import { md3 } from "vuetify/blueprints";
import * as components from "vuetify/components";
import * as directives from "vuetify/directives";
import index from "./index.vue" import "@mdi/font/css/materialdesignicons.min.css";
import router from "./router" import "@fontsource/roboto/latin.css";
import "@unocss/reset/tailwind.css";
const app = createApp(index) import index from "./index.vue";
import router from "./router";
const app = createApp(index);
app.use( app.use(
createVuetify({ createVuetify({
components, components,
directives, directives,
blueprint: md3,
theme: { theme: {
defaultTheme: "original",
themes: { themes: {
light: { original: {
colors: {
primary: "#4a5099", primary: "#4a5099",
secondary: "#2196f3", secondary: "#2196f3",
accent: "#009688", accent: "#009688",
@ -32,10 +40,11 @@ app.use(
} }
} }
} }
}
}) })
) );
app.use(createPinia()) app.use(createPinia());
app.use(router) app.use(router);
app.mount("#app") app.mount("#app");

View File

@ -1,6 +1,5 @@
import { createRouter, createWebHistory } from "vue-router" import { createRouter, createWebHistory } from "vue-router"
import MasterLayout from "@/layouts/master.vue" import MasterLayout from "@/layouts/master.vue"
import LandingPage from "@/views/landing.vue"
const router = createRouter({ const router = createRouter({
history: createWebHistory(import.meta.env.BASE_URL), history: createWebHistory(import.meta.env.BASE_URL),
@ -11,8 +10,8 @@ const router = createRouter({
children: [ children: [
{ {
path: "/", path: "/",
name: "landing", name: "explore",
component: LandingPage component: () => import("@/views/explore.vue")
} }
] ]
} }

View File

@ -0,0 +1,10 @@
declare global {
interface Window {
__LAUNCHPAD_TARGET__?: string
}
}
export async function request(input: string, init?: RequestInit) {
const prefix = window.__LAUNCHPAD_TARGET__ ?? ""
return await fetch(prefix + input, init)
}

View File

@ -0,0 +1,56 @@
import Cookie from "universal-cookie"
import { defineStore } from "pinia"
import { ref } from "vue"
import { request } from "@/scripts/request"
export interface Userinfo {
isReady: boolean
isLoggedIn: boolean
displayName: string
data: any
}
const defaultUserinfo: Userinfo = {
isReady: false,
isLoggedIn: false,
displayName: "Citizen",
data: null
}
export function getAtk(): string {
return new Cookie().get("identity_auth_key")
}
export function checkLoggedIn(): boolean {
return new Cookie().get("identity_auth_key")
}
export const useUserinfo = defineStore("userinfo", () => {
const userinfo = ref(defaultUserinfo)
const isReady = ref(false)
async function readProfiles() {
if (!checkLoggedIn()) {
isReady.value = true;
}
const res = await request("/api/users/me", {
headers: { "Authorization": `Bearer ${getAtk()}` }
});
if (res.status !== 200) {
return;
}
const data = await res.json();
userinfo.value = {
isReady: true,
isLoggedIn: true,
displayName: data["nick"],
data: data
};
}
return { userinfo, isReady, readProfiles }
})

View File

@ -0,0 +1,14 @@
import { request } from "@/scripts/request"
import { defineStore } from "pinia"
import { ref } from "vue"
export const useWellKnown = defineStore("well-known", () => {
const wellKnown = ref({})
async function readWellKnown() {
const res = await request("/.well-known")
wellKnown.value = await res.json()
}
return { wellKnown, readWellKnown }
})

View File

@ -0,0 +1,59 @@
<template>
<v-container class="flex max-md:flex-col gap-3 overflow-auto max-h-[calc(100vh-72px)] no-scrollbar">
<div class="timeline flex-grow-1 mt-[-16px]">
<post-list :loading="loading" :posts="posts" :loader="readMore" />
</div>
<div class="aside sticky top-0 w-full h-fit md:min-w-[280px] md:max-w-[320px]">
<v-card title="Categories">
<v-list density="compact">
</v-list>
</v-card>
</div>
</v-container>
</template>
<script setup lang="ts">
import PostList from "@/components/posts/PostList.vue";
import { reactive, ref } from "vue";
import { request } from "@/scripts/request";
const error = ref<string | null>(null);
const loading = ref(false);
const pagination = reactive({ page: 1, pageSize: 10, total: 0 });
const posts = ref<any[]>([]);
async function readPosts() {
loading.value = true;
const res = await request(`/api/posts?` + new URLSearchParams({
take: pagination.pageSize.toString(),
offset: ((pagination.page - 1) * pagination.pageSize).toString()
}));
if (res.status !== 200) {
loading.value = false;
error.value = await res.text();
} else {
error.value = null;
loading.value = false;
const data = await res.json();
pagination.total = data["count"];
posts.value.push(...data["data"]);
}
}
async function readMore({ done }: any) {
// Reach the end of data
if (pagination.total <= pagination.page * pagination.pageSize) {
done("empty");
return;
}
pagination.page++;
await readPosts();
done("ok");
}
readPosts();
</script>

View File

@ -1,3 +0,0 @@
<template>
<div>Good morning!</div>
</template>

View File

@ -4,6 +4,8 @@
"exclude": ["src/**/__tests__/*"], "exclude": ["src/**/__tests__/*"],
"compilerOptions": { "compilerOptions": {
"composite": true, "composite": true,
"allowJs": true,
"checkJs": true,
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo", "tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo",
"baseUrl": ".", "baseUrl": ".",

View File

@ -12,5 +12,11 @@ export default defineConfig({
alias: { alias: {
"@": fileURLToPath(new URL("./src", import.meta.url)) "@": fileURLToPath(new URL("./src", import.meta.url))
} }
},
server: {
proxy: {
"/.well-known": "http://localhost:8445",
"/api": "http://localhost:8445"
}
} }
}) })