Compare commits

..

7 Commits

Author SHA1 Message Date
tim
1c28201cb8 fix: 侧边栏收起打开 有引导 #848 2025-09-07 14:13:48 +08:00
tim
0e26758585 fix: 首页帖子padding 写为15px 2025-09-07 14:10:04 +08:00
tim
786e60e8e5 fix: 帖子元素 -- 上下间距一样 #846 2025-09-07 14:08:37 +08:00
tim
d94302635a fix: 生产环境新增www 2025-09-07 13:58:22 +08:00
tim
9519f66474 fix: 首页帖子padding可以大一些 比如 20px #845 2025-09-07 13:43:40 +08:00
tim
92ba475f3b fix: 更新分类选择 2025-09-07 13:38:09 +08:00
tim
6fffdb0fd6 feat: 侧边栏按钮样式逻辑修改 2025-09-07 13:20:53 +08:00
13 changed files with 64 additions and 92 deletions

View File

@@ -10,7 +10,6 @@ import java.util.List;
public interface PointHistoryRepository extends JpaRepository<PointHistory, Long> {
List<PointHistory> findByUserOrderByIdDesc(User user);
List<PointHistory> findByUserOrderByIdAsc(User user);
long countByUser(User user);
List<PointHistory> findByUserAndCreatedAtAfterOrderByCreatedAtDesc(User user, LocalDateTime createdAt);

View File

@@ -225,20 +225,17 @@ public class PointService {
*/
public int recalculateUserPoints(User user) {
// 获取用户所有的积分历史记录(由于@Where注解已删除的记录会被自动过滤
List<PointHistory> histories = pointHistoryRepository.findByUserOrderByIdAsc(user);
List<PointHistory> histories = pointHistoryRepository.findByUserOrderByIdDesc(user);
int totalPoints = 0;
for (PointHistory history : histories) {
totalPoints += history.getAmount();
// 重新计算每条历史记录的余额
history.setBalance(totalPoints);
}
// 批量更新历史记录及用户积分
pointHistoryRepository.saveAll(histories);
// 更新用户积分
user.setPoint(totalPoints);
userRepository.save(user);
return totalPoints;
}

View File

@@ -1,7 +1,6 @@
package com.openisle.service;
import com.openisle.exception.FieldException;
import org.apache.commons.lang3.math.NumberUtils;
import org.springframework.stereotype.Service;
/**
@@ -18,11 +17,6 @@ public class UsernameValidator {
if (username == null || username.isEmpty()) {
throw new FieldException("username", "Username cannot be empty");
}
if (NumberUtils.isDigits(username)) {
throw new FieldException("username", "Username cannot be pure number");
}
}
}

View File

@@ -1,67 +0,0 @@
package com.openisle.service;
import com.openisle.model.PointHistory;
import com.openisle.model.PointHistoryType;
import com.openisle.model.Role;
import com.openisle.model.User;
import com.openisle.repository.PointHistoryRepository;
import com.openisle.repository.UserRepository;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
import org.springframework.context.annotation.Import;
import java.time.LocalDateTime;
import static org.junit.jupiter.api.Assertions.assertEquals;
@DataJpaTest
@Import(PointService.class)
class PointServiceRecalculateUserPointsTest {
@Autowired
private PointService pointService;
@Autowired
private UserRepository userRepository;
@Autowired
private PointHistoryRepository pointHistoryRepository;
@Test
void recalculatesBalanceAfterDeletion() {
User user = new User();
user.setUsername("u");
user.setEmail("u@example.com");
user.setPassword("p");
user.setRole(Role.USER);
userRepository.save(user);
PointHistory h1 = new PointHistory();
h1.setUser(user);
h1.setType(PointHistoryType.POST);
h1.setAmount(30);
h1.setBalance(30);
h1.setCreatedAt(LocalDateTime.now().minusMinutes(2));
pointHistoryRepository.save(h1);
PointHistory h2 = new PointHistory();
h2.setUser(user);
h2.setType(PointHistoryType.COMMENT);
h2.setAmount(10);
h2.setBalance(40);
h2.setCreatedAt(LocalDateTime.now().minusMinutes(1));
pointHistoryRepository.save(h2);
user.setPoint(40);
userRepository.save(user);
pointHistoryRepository.delete(h1);
int total = pointService.recalculateUserPoints(user);
assertEquals(10, total);
assertEquals(10, userRepository.findById(user.getId()).orElseThrow().getPoint());
assertEquals(10, pointHistoryRepository.findById(h2.getId()).orElseThrow().getBalance());
}
}

View File

@@ -1,10 +1,10 @@
; 生产环境后端
NUXT_PUBLIC_API_BASE_URL=https://open-isle.com
NUXT_PUBLIC_API_BASE_URL=https://www.open-isle.com
; 正式环境/生产环境
NUXT_PUBLIC_WEBSITE_BASE_URL=https://open-isle.com
NUXT_PUBLIC_WEBSITE_BASE_URL=https://www.open-isle.com
; 生产环境ws后端
NUXT_PUBLIC_WEBSOCKET_URL=https://open-isle.com/websocket
NUXT_PUBLIC_WEBSOCKET_URL=https://www.open-isle.com/websocket
NUXT_PUBLIC_GOOGLE_CLIENT_ID=777830451304-nt8afkkap18gui4f9entcha99unal744.apps.googleusercontent.com
NUXT_PUBLIC_GITHUB_CLIENT_ID=Ov23liVkO1NPAX5JyWxJ

View File

@@ -2,11 +2,12 @@
<div class="article-category-container" v-if="category">
<div class="article-info-item" @click="gotoCategory">
<BaseImage
v-if="category.smallIcon"
v-if="isImageIcon(category.smallIcon)"
class="article-info-item-img"
:src="category.smallIcon"
:alt="category.name"
/>
<component v-else :is="category.smallIcon || category.icon" class="article-info-item-img" />
<div class="article-info-item-text">{{ category.name }}</div>
</div>
</div>
@@ -22,6 +23,11 @@ const gotoCategory = async () => {
const value = encodeURIComponent(props.category.id ?? props.category.name)
await navigateTo({ path: '/', query: { category: value } }, { replace: true })
}
const isImageIcon = (icon) => {
if (!icon) return false
return /^https?:\/\//.test(icon) || icon.startsWith('/')
}
</script>
<style scoped>

View File

@@ -7,11 +7,17 @@
@click="gotoTag(tag)"
>
<BaseImage
v-if="tag.smallIcon"
v-if="isImageIcon(tag.smallIcon)"
class="article-info-item-img"
:src="tag.smallIcon"
:alt="tag.name"
/>
<component
v-else-if="tag.smallIcon || tag.icon"
:is="tag.smallIcon || tag.icon"
class="article-info-item-img"
/>
<tag-one v-else class="article-info-item-img" />
<div class="article-info-item-text">{{ tag.name }}</div>
</div>
</div>
@@ -26,6 +32,11 @@ const gotoTag = async (tag) => {
const value = encodeURIComponent(tag.id ?? tag.name)
await navigateTo({ path: '/', query: { tags: value } }, { replace: true })
}
const isImageIcon = (icon) => {
if (!icon) return false
return /^https?:\/\//.test(icon) || icon.startsWith('/')
}
</script>
<style scoped>

View File

@@ -15,7 +15,7 @@
class="option-icon"
:alt="option.name"
/>
<!-- <i v-else :class="['option-icon', option.icon]"></i> -->
<component v-else :is="option.smallIcon || option.icon" class="option-icon" />
</template>
<span>{{ option.name }}</span>
<span class="option-count" v-if="option.count > 0"> x {{ option.count }}</span>

View File

@@ -404,7 +404,6 @@ const handleContentClick = (e) => {
margin-left: 10px;
margin-right: 10px;
opacity: 0.5;
transform: scaleX(-1);
}
.reply-user-name {

View File

@@ -4,7 +4,9 @@
<div class="header-content-left">
<div v-if="showMenuBtn" class="menu-btn-wrapper">
<button class="menu-btn" ref="menuBtn" @click="$emit('toggle-menu')">
<application-menu class="micon"></application-menu>
<ToolTip content="展开/收起菜单" placement="bottom">
<application-menu class="micon"></application-menu>
</ToolTip>
</button>
<span
v-if="isMobile && (unreadMessageCount > 0 || hasChannelUnread)"

View File

@@ -122,6 +122,11 @@
class="section-item-icon"
:alt="t.name"
/>
<component
v-else-if="t.smallIcon || t.icon"
:is="t.smallIcon || t.icon"
class="section-item-icon"
/>
<tag-one v-else class="section-item-icon" />
<span class="section-item-text"
>{{ t.name }} <span class="section-item-text-count">x {{ t.count }}</span></span

View File

@@ -529,7 +529,7 @@ const sanitizeDescription = (text) => stripMarkdown(text)
}
.article-item-title {
margin-top: 10px;
margin-top: 15px;
font-size: 18px;
text-decoration: none;
color: var(--text-color);
@@ -558,7 +558,7 @@ const sanitizeDescription = (text) => stripMarkdown(text)
.article-item-description {
max-width: 100%;
margin-top: 10px;
margin-top: 5px;
font-size: 13px;
color: rgba(140, 140, 140, 0.888);
display: -webkit-box;
@@ -605,6 +605,7 @@ const sanitizeDescription = (text) => stripMarkdown(text)
display: flex;
flex-direction: column;
align-items: flex-start;
padding-bottom: 5px;
}
.article-member-avatars-container {
@@ -719,10 +720,15 @@ const sanitizeDescription = (text) => stripMarkdown(text)
}
.article-item-title {
margin-top: 10px;
font-size: 16px;
font-weight: bold;
}
.article-main-container {
padding-bottom: 0px;
}
.article-item-description {
margin-top: 2px;
font-size: 10px;

View File

@@ -63,6 +63,16 @@ import {
History,
Lightning,
PeoplesTwo,
Code,
GoodTwo,
Twitter,
Bitcoin,
Fire,
Communication,
WaterLevel,
RobotOne,
Server,
Protection,
} from '@icon-park/vue-next'
export default defineNuxtPlugin((nuxtApp) => {
@@ -129,4 +139,14 @@ export default defineNuxtPlugin((nuxtApp) => {
nuxtApp.vueApp.component('HistoryIcon', History)
nuxtApp.vueApp.component('Lightning', Lightning)
nuxtApp.vueApp.component('PeoplesTwo', PeoplesTwo)
nuxtApp.vueApp.component('CodeIcon', Code)
nuxtApp.vueApp.component('GoodTwo', GoodTwo)
nuxtApp.vueApp.component('Twitter', Twitter)
nuxtApp.vueApp.component('Bitcoin', Bitcoin)
nuxtApp.vueApp.component('Fire', Fire)
nuxtApp.vueApp.component('Communication', Communication)
nuxtApp.vueApp.component('WaterLevel', WaterLevel)
nuxtApp.vueApp.component('RobotOne', RobotOne)
nuxtApp.vueApp.component('ServerIcon', Server)
nuxtApp.vueApp.component('Protection', Protection)
})